opal-browser 0.1.0.beta1 → 0.3.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (257) hide show
  1. checksums.yaml +5 -5
  2. data/.github/workflows/build.yml +95 -0
  3. data/.gitignore +3 -0
  4. data/.yardopts +1 -1
  5. data/Gemfile +22 -3
  6. data/LICENSE +20 -0
  7. data/README.md +200 -20
  8. data/Rakefile +29 -1
  9. data/config.ru +20 -2
  10. data/docs/polyfills.md +24 -0
  11. data/examples/2048/Gemfile +6 -0
  12. data/examples/2048/README.md +13 -0
  13. data/examples/2048/app/application.rb +169 -0
  14. data/examples/2048/config.ru +9 -0
  15. data/examples/canvas/Gemfile +6 -0
  16. data/examples/canvas/README.md +9 -0
  17. data/examples/canvas/app/application.rb +55 -0
  18. data/examples/canvas/config.ru +9 -0
  19. data/examples/component/Gemfile +6 -0
  20. data/examples/component/README.md +10 -0
  21. data/examples/component/app/application.rb +66 -0
  22. data/examples/component/config.ru +9 -0
  23. data/examples/integrations/README.md +24 -0
  24. data/examples/integrations/dynamic-rack-opal-sprockets-server/Gemfile +6 -0
  25. data/examples/integrations/dynamic-rack-opal-sprockets-server/README.md +16 -0
  26. data/examples/integrations/dynamic-rack-opal-sprockets-server/app/application.rb +6 -0
  27. data/examples/integrations/dynamic-rack-opal-sprockets-server/config.ru +9 -0
  28. data/examples/integrations/dynamic-roda-roda-sprockets/.gitignore +1 -0
  29. data/examples/integrations/dynamic-roda-roda-sprockets/Gemfile +7 -0
  30. data/examples/integrations/dynamic-roda-roda-sprockets/README.md +22 -0
  31. data/examples/integrations/dynamic-roda-roda-sprockets/Rakefile +4 -0
  32. data/examples/integrations/dynamic-roda-roda-sprockets/app/application.rb +6 -0
  33. data/examples/integrations/dynamic-roda-roda-sprockets/app.rb +32 -0
  34. data/examples/integrations/dynamic-roda-roda-sprockets/config.ru +3 -0
  35. data/examples/integrations/dynamic-roda-tilt/.gitignore +1 -0
  36. data/examples/integrations/dynamic-roda-tilt/Gemfile +8 -0
  37. data/examples/integrations/dynamic-roda-tilt/README.md +17 -0
  38. data/examples/integrations/dynamic-roda-tilt/Rakefile +6 -0
  39. data/examples/integrations/dynamic-roda-tilt/app/application.rb +6 -0
  40. data/examples/integrations/dynamic-roda-tilt/app.rb +50 -0
  41. data/examples/integrations/dynamic-roda-tilt/config.ru +3 -0
  42. data/examples/integrations/dynamic-sinatra-opal-sprockets-server/Gemfile +7 -0
  43. data/examples/integrations/dynamic-sinatra-opal-sprockets-server/README.md +16 -0
  44. data/examples/integrations/dynamic-sinatra-opal-sprockets-server/app/application.rb +6 -0
  45. data/examples/integrations/dynamic-sinatra-opal-sprockets-server/config.ru +29 -0
  46. data/examples/integrations/static-bash/.gitignore +2 -0
  47. data/examples/integrations/static-bash/Gemfile +3 -0
  48. data/examples/integrations/static-bash/README.md +8 -0
  49. data/examples/integrations/static-bash/app/application.rb +6 -0
  50. data/examples/integrations/static-bash/build.sh +4 -0
  51. data/examples/integrations/static-bash/index.html +10 -0
  52. data/examples/integrations/static-bash-opal-parser/.gitignore +3 -0
  53. data/examples/integrations/static-bash-opal-parser/Gemfile +3 -0
  54. data/examples/integrations/static-bash-opal-parser/README.md +10 -0
  55. data/examples/integrations/static-bash-opal-parser/build.sh +4 -0
  56. data/examples/integrations/static-bash-opal-parser/index.html +19 -0
  57. data/examples/integrations/static-rake/.gitignore +1 -0
  58. data/examples/integrations/static-rake/Gemfile +4 -0
  59. data/examples/integrations/static-rake/README.md +7 -0
  60. data/examples/integrations/static-rake/Rakefile +10 -0
  61. data/examples/integrations/static-rake/app/application.rb +6 -0
  62. data/examples/integrations/static-rake/index.html +9 -0
  63. data/examples/integrations/static-rake-guard/.gitignore +1 -0
  64. data/examples/integrations/static-rake-guard/Gemfile +6 -0
  65. data/examples/integrations/static-rake-guard/Guardfile +3 -0
  66. data/examples/integrations/static-rake-guard/README.md +10 -0
  67. data/examples/integrations/static-rake-guard/Rakefile +10 -0
  68. data/examples/integrations/static-rake-guard/app/application.rb +6 -0
  69. data/examples/integrations/static-rake-guard/index.html +9 -0
  70. data/examples/svg/.gitignore +1 -0
  71. data/examples/svg/Gemfile +4 -0
  72. data/examples/svg/README.md +7 -0
  73. data/examples/svg/Rakefile +10 -0
  74. data/examples/svg/app/application.rb +11 -0
  75. data/examples/svg/index.html +17 -0
  76. data/examples/svg/index.svg +6 -0
  77. data/index.html.erb +24 -0
  78. data/lib/opal-browser.rb +1 -0
  79. data/opal/browser/animation_frame.rb +92 -10
  80. data/opal/browser/audio/node.rb +121 -0
  81. data/opal/browser/audio/param_schedule.rb +43 -0
  82. data/opal/browser/audio.rb +66 -0
  83. data/opal/browser/blob.rb +94 -0
  84. data/opal/browser/canvas/data.rb +2 -12
  85. data/opal/browser/canvas/gradient.rb +1 -11
  86. data/opal/browser/canvas/style.rb +3 -11
  87. data/opal/browser/canvas/text.rb +1 -11
  88. data/opal/browser/canvas.rb +86 -28
  89. data/opal/browser/console.rb +6 -38
  90. data/opal/browser/cookies.rb +90 -27
  91. data/opal/browser/crypto.rb +79 -0
  92. data/opal/browser/css/declaration.rb +1 -6
  93. data/opal/browser/css/rule.rb +1 -1
  94. data/opal/browser/css/style_sheet.rb +2 -2
  95. data/opal/browser/css.rb +23 -7
  96. data/opal/browser/database/sql.rb +193 -0
  97. data/opal/browser/delay.rb +94 -0
  98. data/opal/browser/dom/attribute.rb +16 -9
  99. data/opal/browser/dom/builder.rb +35 -25
  100. data/opal/browser/dom/character_data.rb +43 -7
  101. data/opal/browser/dom/document.rb +171 -37
  102. data/opal/browser/dom/document_fragment.rb +18 -0
  103. data/opal/browser/dom/document_or_shadow_root.rb +19 -0
  104. data/opal/browser/dom/element/attributes.rb +111 -0
  105. data/opal/browser/dom/element/button.rb +31 -0
  106. data/opal/browser/dom/element/custom.rb +177 -0
  107. data/opal/browser/dom/element/data.rb +82 -0
  108. data/opal/browser/dom/element/editable.rb +47 -0
  109. data/opal/browser/dom/element/form.rb +38 -0
  110. data/opal/browser/dom/element/iframe.rb +37 -0
  111. data/opal/browser/dom/element/image.rb +25 -0
  112. data/opal/browser/dom/element/input.rb +48 -1
  113. data/opal/browser/dom/element/media.rb +17 -0
  114. data/opal/browser/dom/element/offset.rb +32 -10
  115. data/opal/browser/dom/element/position.rb +11 -2
  116. data/opal/browser/dom/element/scroll.rb +139 -20
  117. data/opal/browser/dom/element/select.rb +42 -0
  118. data/opal/browser/dom/element/size.rb +46 -0
  119. data/opal/browser/dom/element/template.rb +11 -0
  120. data/opal/browser/dom/element/textarea.rb +26 -0
  121. data/opal/browser/dom/element.rb +496 -168
  122. data/opal/browser/dom/mutation_observer.rb +69 -9
  123. data/opal/browser/dom/node.rb +270 -83
  124. data/opal/browser/dom/node_set.rb +74 -41
  125. data/opal/browser/dom/shadow_root.rb +12 -0
  126. data/opal/browser/dom/text.rb +18 -3
  127. data/opal/browser/dom.rb +40 -18
  128. data/opal/browser/effects.rb +180 -3
  129. data/opal/browser/event/all.rb +26 -0
  130. data/opal/browser/event/animation.rb +40 -0
  131. data/opal/browser/{dom/event → event}/audio_processing.rb +10 -6
  132. data/opal/browser/event/base.rb +461 -0
  133. data/opal/browser/event/before_unload.rb +17 -0
  134. data/opal/browser/event/clipboard.rb +37 -0
  135. data/opal/browser/event/close.rb +49 -0
  136. data/opal/browser/event/composition.rb +52 -0
  137. data/opal/browser/event/custom.rb +65 -0
  138. data/opal/browser/event/data_transfer.rb +95 -0
  139. data/opal/browser/event/device_light.rb +25 -0
  140. data/opal/browser/{dom/event → event}/device_motion.rb +21 -6
  141. data/opal/browser/event/device_orientation.rb +50 -0
  142. data/opal/browser/{dom/event → event}/device_proximity.rb +10 -6
  143. data/opal/browser/event/drag.rb +123 -0
  144. data/opal/browser/event/focus.rb +41 -0
  145. data/opal/browser/event/gamepad.rb +62 -0
  146. data/opal/browser/{dom/event → event}/hash_change.rb +10 -6
  147. data/opal/browser/event/keyboard.rb +128 -0
  148. data/opal/browser/event/message.rb +72 -0
  149. data/opal/browser/{dom/event → event}/mouse.rb +37 -32
  150. data/opal/browser/event/page_transition.rb +25 -0
  151. data/opal/browser/event/pop_state.rb +35 -0
  152. data/opal/browser/event/progress.rb +45 -0
  153. data/opal/browser/event/sensor.rb +17 -0
  154. data/opal/browser/{dom/event → event}/storage.rb +10 -6
  155. data/opal/browser/{dom/event → event}/touch.rb +14 -21
  156. data/opal/browser/event/ui.rb +38 -0
  157. data/opal/browser/{dom/event → event}/wheel.rb +6 -4
  158. data/opal/browser/event.rb +163 -0
  159. data/opal/browser/event_source.rb +7 -4
  160. data/opal/browser/form_data.rb +225 -0
  161. data/opal/browser/history.rb +53 -21
  162. data/opal/browser/http/binary.rb +1 -0
  163. data/opal/browser/http/headers.rb +21 -2
  164. data/opal/browser/http/request.rb +83 -55
  165. data/opal/browser/http/response.rb +5 -1
  166. data/opal/browser/http.rb +47 -9
  167. data/opal/browser/immediate.rb +128 -10
  168. data/opal/browser/interval.rb +41 -23
  169. data/opal/browser/location.rb +20 -4
  170. data/opal/browser/navigator.rb +136 -13
  171. data/opal/browser/polyfill/visual_viewport.rb +216 -0
  172. data/opal/browser/screen.rb +34 -8
  173. data/opal/browser/setup/base.rb +6 -0
  174. data/opal/browser/setup/full.rb +13 -0
  175. data/opal/browser/setup/large.rb +17 -0
  176. data/opal/browser/setup/mini.rb +8 -0
  177. data/opal/browser/setup/traditional.rb +10 -0
  178. data/opal/browser/socket.rb +16 -8
  179. data/opal/browser/storage.rb +155 -52
  180. data/opal/browser/support.rb +299 -0
  181. data/opal/browser/utils.rb +116 -18
  182. data/opal/browser/version.rb +1 -1
  183. data/opal/browser/visual_viewport.rb +39 -0
  184. data/opal/browser/window/size.rb +47 -9
  185. data/opal/browser/window/view.rb +37 -4
  186. data/opal/browser/window.rb +46 -26
  187. data/opal/browser.rb +1 -10
  188. data/opal/opal-browser.rb +1 -0
  189. data/opal-browser.gemspec +10 -12
  190. data/spec/database/sql_spec.rb +139 -0
  191. data/spec/delay_spec.rb +41 -0
  192. data/spec/dom/attribute_spec.rb +49 -0
  193. data/spec/dom/builder_spec.rb +36 -19
  194. data/spec/dom/document_spec.rb +28 -6
  195. data/spec/dom/element/attributes_spec.rb +52 -0
  196. data/spec/dom/element/custom_spec.rb +106 -0
  197. data/spec/dom/element/subclass_spec.rb +144 -0
  198. data/spec/dom/element_spec.rb +184 -7
  199. data/spec/dom/mutation_observer_spec.rb +13 -9
  200. data/spec/dom/node_set_spec.rb +44 -0
  201. data/spec/dom/node_spec.rb +87 -27
  202. data/spec/dom_spec.rb +19 -9
  203. data/spec/event_source_spec.rb +18 -15
  204. data/spec/{dom/event_spec.rb → event_spec.rb} +55 -26
  205. data/spec/history_spec.rb +32 -19
  206. data/spec/http_spec.rb +25 -36
  207. data/spec/immediate_spec.rb +10 -7
  208. data/spec/interval_spec.rb +59 -0
  209. data/spec/native_cached_wrapper_spec.rb +46 -0
  210. data/spec/runner.rb +107 -0
  211. data/spec/socket_spec.rb +18 -14
  212. data/spec/spec_helper.rb +2 -4
  213. data/spec/spec_helper_promise.rb.erb +25 -0
  214. data/spec/storage_spec.rb +7 -7
  215. data/spec/wgxpath.install.js +49 -0
  216. data/spec/window_spec.rb +2 -2
  217. metadata +181 -93
  218. data/opal/browser/compatibility/animation_frame.rb +0 -93
  219. data/opal/browser/compatibility/dom/document/window.rb +0 -15
  220. data/opal/browser/compatibility/dom/element/css.rb +0 -15
  221. data/opal/browser/compatibility/dom/element/matches.rb +0 -31
  222. data/opal/browser/compatibility/dom/element/offset.rb +0 -20
  223. data/opal/browser/compatibility/dom/element/scroll.rb +0 -25
  224. data/opal/browser/compatibility/dom/element/style.rb +0 -15
  225. data/opal/browser/compatibility/dom/mutation_observer.rb +0 -47
  226. data/opal/browser/compatibility/http/request.rb +0 -15
  227. data/opal/browser/compatibility/immediate.rb +0 -107
  228. data/opal/browser/compatibility/window/scroll.rb +0 -27
  229. data/opal/browser/compatibility/window/size.rb +0 -13
  230. data/opal/browser/compatibility/window/view.rb +0 -13
  231. data/opal/browser/compatibility.rb +0 -59
  232. data/opal/browser/dom/compatibility.rb +0 -8
  233. data/opal/browser/dom/event/animation.rb +0 -26
  234. data/opal/browser/dom/event/base.rb +0 -207
  235. data/opal/browser/dom/event/before_unload.rb +0 -13
  236. data/opal/browser/dom/event/clipboard.rb +0 -26
  237. data/opal/browser/dom/event/close.rb +0 -35
  238. data/opal/browser/dom/event/composition.rb +0 -38
  239. data/opal/browser/dom/event/custom.rb +0 -30
  240. data/opal/browser/dom/event/device_light.rb +0 -21
  241. data/opal/browser/dom/event/device_orientation.rb +0 -36
  242. data/opal/browser/dom/event/drag.rb +0 -113
  243. data/opal/browser/dom/event/focus.rb +0 -23
  244. data/opal/browser/dom/event/gamepad.rb +0 -47
  245. data/opal/browser/dom/event/keyboard.rb +0 -93
  246. data/opal/browser/dom/event/message.rb +0 -50
  247. data/opal/browser/dom/event/page_transition.rb +0 -21
  248. data/opal/browser/dom/event/pop_state.rb +0 -21
  249. data/opal/browser/dom/event/progress.rb +0 -31
  250. data/opal/browser/dom/event/sensor.rb +0 -13
  251. data/opal/browser/dom/event/ui.rb +0 -22
  252. data/opal/browser/dom/event.rb +0 -240
  253. data/opal/browser/http/compatibility.rb +0 -1
  254. data/opal/browser/http/parameters.rb +0 -8
  255. data/opal/browser/timeout.rb +0 -60
  256. data/opal/browser/window/compatibility.rb +0 -3
  257. data/opal/browser/window/scroll.rb +0 -49
@@ -0,0 +1,47 @@
1
+ module Browser; module DOM
2
+
3
+ class Element < Node
4
+ # @!attribute editable
5
+ # @return [Boolean?] the value of contentEditable for this element
6
+ def editable
7
+ case `#@native.contentEditable`
8
+ when "true"
9
+ true
10
+ when "false"
11
+ false
12
+ when "inherit"
13
+ nil
14
+ end
15
+ end
16
+
17
+ def editable=(value)
18
+ value = case value
19
+ when true
20
+ "true"
21
+ when false
22
+ "false"
23
+ when nil
24
+ "inherit"
25
+ end
26
+ `#@native.contentEditable = #{value}`
27
+ end
28
+
29
+ def editable?
30
+ `#@native.isContentEditable`
31
+ end
32
+
33
+ # Execute a contentEditable command
34
+ def edit(command, value=nil)
35
+ command = command.gsub(/_./) { |i| i[1].upcase }
36
+
37
+ focus
38
+
39
+ if value
40
+ `#{document}.native.execCommand(#{command}, false, #{value})`
41
+ else
42
+ `#{document}.native.execCommand(#{command}, false)`
43
+ end
44
+ end
45
+ end
46
+
47
+ end; end
@@ -0,0 +1,38 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Form < Element
4
+ def_selector "form"
5
+
6
+ # Capture the content of this form to a new {FormData} object,
7
+ #
8
+ # @return [FormData]
9
+ def form_data
10
+ FormData.create(self)
11
+ end
12
+
13
+ # Submit a form. This will fire a submit event.
14
+ def submit
15
+ `#@native.submit()`
16
+ end
17
+
18
+ # Reset a form. This will fire a reset event.
19
+ def reset
20
+ `#@native.reset()`
21
+ end
22
+
23
+ alias_native :action
24
+ alias_native :action=
25
+ alias_native :method
26
+ alias_native :method=
27
+ alias_native :target
28
+ alias_native :target=
29
+ alias_native :encoding
30
+ alias_native :encoding=
31
+
32
+ # Return a NodeSet containing all form controls belonging to this form element.
33
+ def controls
34
+ NodeSet[Native::Array.new(`#@native.elements`)]
35
+ end
36
+ end
37
+
38
+ end; end; end
@@ -0,0 +1,37 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Iframe < Element
4
+ def_selector "iframe"
5
+
6
+ # @!attribute src
7
+ # @return [String] the URL of the page to embed
8
+ alias_native :src
9
+ alias_native :src=
10
+
11
+ # @!attribute [r] content_window
12
+ # @return [Window] window of content of this iframe
13
+ def content_window
14
+ Browser::Window.new(`#@native.contentWindow`)
15
+ end
16
+
17
+ # @!attribute [r] content_document
18
+ # @return [Document] document of content of this iframe
19
+ def content_document
20
+ DOM(`#@native.contentDocument || #@native.contentWindow.document`)
21
+ end
22
+
23
+ # Send a message to the iframe content's window.
24
+ #
25
+ # @param message [String] the message
26
+ # @param options [Hash] optional `to: target`
27
+ def send(message, options={})
28
+ content_window.send(message, options)
29
+ end
30
+ end
31
+
32
+ # Object is not an iframe, but acts the same.
33
+ class Object < Iframe
34
+ def_selector "object"
35
+ end
36
+
37
+ end; end; end
@@ -0,0 +1,25 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Image < Element
4
+ def_selector "img"
5
+
6
+ def complete?
7
+ `#@native.complete`
8
+ end
9
+
10
+ def cross?
11
+ `#@native.crossOrigin`
12
+ end
13
+
14
+ def height
15
+ `#@native.naturalHeight`
16
+ end
17
+
18
+ def width
19
+ `#@native.naturalWidth`
20
+ end
21
+ end
22
+
23
+ Img = Image
24
+
25
+ end; end; end
@@ -1,17 +1,64 @@
1
1
  module Browser; module DOM; class Element < Node
2
2
 
3
3
  class Input < Element
4
+ def_selector "input"
5
+
4
6
  def value
5
- `#@native.value`
7
+ %x{
8
+ if (#@native.value == "") {
9
+ return nil;
10
+ }
11
+ else {
12
+ return #@native.value;
13
+ }
14
+ }
6
15
  end
7
16
 
8
17
  def value=(value)
9
18
  `#@native.value = #{value}`
10
19
  end
11
20
 
21
+ def name_
22
+ `#@native.name`
23
+ end
24
+
25
+ def type
26
+ `#@native.type`
27
+ end
28
+
29
+ def checked?
30
+ `#@native.checked`
31
+ end
32
+
33
+ def check!
34
+ `#@native.checked = 'checked'`
35
+ end
36
+
37
+ def uncheck!
38
+ `#@native.checked = ''`
39
+ end
40
+
41
+ def enabled?
42
+ `#@native.enabled`
43
+ end
44
+
45
+ def disable!
46
+ `#@native.disabled = 'disabled'`
47
+ end
48
+
49
+ def enable!
50
+ `#@native.disabled = ''`
51
+ end
52
+
12
53
  def clear
13
54
  `#@native.value = ''`
14
55
  end
56
+
57
+ # @!attribute [r] files
58
+ # @return [Array<File>] list of files attached to this {Input}
59
+ def files
60
+ Native::Array.new(`#@native.files`).map { |f| File.new(f.to_n) }
61
+ end
15
62
  end
16
63
 
17
64
  end; end; end
@@ -0,0 +1,17 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Media < Element
4
+ def play
5
+ `#@native.play()`
6
+ end
7
+ end
8
+
9
+ class Video < Media
10
+ def_selector "video"
11
+ end
12
+
13
+ class Audio < Media
14
+ def_selector "audio"
15
+ end
16
+
17
+ end; end; end
@@ -1,6 +1,9 @@
1
1
  module Browser; module DOM; class Element < Node
2
2
 
3
3
  class Offset
4
+ attr_reader :element
5
+
6
+ # @private
4
7
  def initialize(element)
5
8
  @element = element
6
9
  @native = element.to_n
@@ -10,6 +13,8 @@ class Offset
10
13
  DOM(`#@native.offsetParent || #{@element.document.root.to_n}`)
11
14
  end
12
15
 
16
+ # @!attribute x
17
+ # @return [Integer]
13
18
  def x
14
19
  get.x
15
20
  end
@@ -18,6 +23,8 @@ class Offset
18
23
  set value, nil
19
24
  end
20
25
 
26
+ # @!attribute y
27
+ # @return [Integer]
21
28
  def y
22
29
  get.y
23
30
  end
@@ -26,18 +33,33 @@ class Offset
26
33
  set nil, value
27
34
  end
28
35
 
29
- def get
30
- doc = @element.document
31
- root = doc.root.to_n
32
- win = doc.window.to_n
36
+ if Browser.supports? 'Element.getBoundingClientRect'
37
+ def get
38
+ doc = @element.document
39
+ root = doc.root.to_n
40
+ win = doc.window.to_n
41
+
42
+ %x{
43
+ var box = #@native.getBoundingClientRect(),
44
+ y = box.top + (#{win}.pageYOffset || #{root}.scrollTop) - (#{root}.clientTop || 0),
45
+ x = box.left + (#{win}.pageXOffset || #{root}.scrollLeft) - (#{root}.clientLeft || 0);
46
+ }
33
47
 
34
- %x{
35
- var box = #@native.getBoundingClientRect(),
36
- y = box.top + (#{win}.pageYOffset || #{root}.scrollTop) - (#{root}.clientTop || 0),
37
- x = box.left + (#{win}.pageXOffset || #{root}.scrollLeft) - (#{root}.clientLeft || 0);
38
- }
48
+ Browser::Position.new(`x`, `y`)
49
+ end
50
+ else
51
+ def get
52
+ doc = document
53
+ root = doc.root.to_n
54
+ win = doc.window.to_n
39
55
 
40
- Browser::Position.new(`x`, `y`)
56
+ %x{
57
+ var y = (#{win}.pageYOffset || #{root}.scrollTop) - (#{root}.clientTop || 0),
58
+ x = (#{win}.pageXOffset || #{root}.scrollLeft) - (#{root}.clientLeft || 0);
59
+ }
60
+
61
+ Browser::Position.new(`x`, `y`)
62
+ end
41
63
  end
42
64
 
43
65
  def set(*value)
@@ -1,11 +1,15 @@
1
1
  module Browser; module DOM; class Element < Node
2
2
 
3
3
  class Position
4
+ attr_reader :element
5
+
6
+ # @private
4
7
  def initialize(element)
5
8
  @element = element
6
9
  @native = element.to_n
7
10
  end
8
11
 
12
+ # @private
9
13
  def get
10
14
  offset = @element.offset
11
15
  position = offset.get
@@ -21,14 +25,19 @@ class Position
21
25
  parent_offset.y += parent.style['border-left-width'].to_i
22
26
  end
23
27
 
24
- Browser::Position.new(position.x - parent_offset.x - @element.style['margin-left'].to_i,
25
- position.y - parent_offset.y - @element.style['margin-top'].to_i)
28
+ Browser::Position.new(
29
+ position.x - parent_offset.x - @element.style['margin-left'].to_i,
30
+ position.y - parent_offset.y - @element.style['margin-top'].to_i)
26
31
  end
27
32
 
33
+ # @!attribute [r] x
34
+ # @return [Integer] the position of the element on the x axis
28
35
  def x
29
36
  get.x
30
37
  end
31
38
 
39
+ # @!attribute [r] y
40
+ # @return [Integer] the position of the element on the y axis
32
41
  def y
33
42
  get.y
34
43
  end
@@ -1,48 +1,167 @@
1
1
  module Browser; module DOM; class Element < Node
2
2
 
3
+ # @todo Consider using the new interfaces which allow for optional
4
+ # smooth transitions.
3
5
  class Scroll
6
+ attr_reader :element
7
+
8
+ # @private
4
9
  def initialize(element)
5
10
  @element = element
6
11
  @native = element.to_n
12
+
13
+ # Portable support for Window#scroll and Document#scroll
14
+ @scrolling_native = @native
15
+ if [Document, Window].include?(@element.class)
16
+ # If we are a window, let's become a document first.
17
+ if defined? `#@scrolling_native.document`
18
+ @scrolling_native = `#@scrolling_native.document`
19
+ end
20
+ # There were slight disagreements in the past which element
21
+ # should we handle.
22
+ if defined? `#@scrolling_native.documentElement.scrollTop`
23
+ @scrolling_native = `#@scrolling_native.documentElement`
24
+ elsif defined? `#@scrolling_native.body.scrollTop`
25
+ @scrolling_native = `#@scrolling_native.body`
26
+ end
27
+ end
7
28
  end
8
29
 
9
- def position
10
- Browser::Position.new(`#@native.scrollLeft`, `#@native.scrollTop`)
30
+ # @overload to(x, y)
31
+ #
32
+ # Scroll to the given x and y.
33
+ #
34
+ # @param x [Integer] scroll to x on the x axis
35
+ # @param y [Integer] scroll to y on the y axis
36
+ #
37
+ # @overload to(hash)
38
+ #
39
+ # Scroll to the given x and y.
40
+ #
41
+ # @param hash [Hash] the descriptor
42
+ #
43
+ # @option hash [Integer] :x scroll to x on the x axis
44
+ # @option hash [Integer] :y scroll to y on the y axis
45
+ #
46
+ # @overload to(symbol)
47
+ #
48
+ # Scroll to :top or to :bottom
49
+ #
50
+ # @param symbol [Symbol] either :top or :bottom
51
+ def to(*args)
52
+ x, y = nil, nil
53
+ case args.first
54
+ when Hash
55
+ x = args.first[:x]
56
+ y = args.first[:y]
57
+ when :top
58
+ y = 0
59
+ when :bottom
60
+ y = 99999999
61
+ else
62
+ x, y = args
63
+ end
64
+
65
+ set(x, y) if x || y
66
+
67
+ self
11
68
  end
12
69
 
13
- def x
14
- position.x
70
+ # @overload by(x, y)
71
+ #
72
+ # Scroll by the given x and y.
73
+ #
74
+ # @param x [Integer] scroll by x on the x axis
75
+ # @param y [Integer] scroll by y on the y axis
76
+ #
77
+ # @overload by(hash)
78
+ #
79
+ # Scroll by the given x and y.
80
+ #
81
+ # @param hash [Hash] the descriptor
82
+ #
83
+ # @option hash [Integer] :x scroll by x on the x axis
84
+ # @option hash [Integer] :y scroll by y on the y axis
85
+ def by(*args)
86
+ case args.first
87
+ when Hash
88
+ x = args.first[:x] || 0
89
+ y = args.first[:y] || 0
90
+ else
91
+ x, y = args
92
+ end
93
+
94
+ set_by(x, y)
95
+
96
+ self
15
97
  end
16
98
 
17
- def y
18
- position.y
99
+ if Browser.supports? 'Element.scrollBy'
100
+ private def set_by(x, y)
101
+ `#@scrolling_native.scrollBy(#{x}, #{y})`
102
+ end
103
+ else
104
+ private def set_by(x, y)
105
+ set(self.x + x, self.y + y)
106
+ end
19
107
  end
20
108
 
21
- def to(what)
22
- x = what[:x] || self.x
23
- y = what[:y] || self.y
109
+ if Browser.supports? 'Element.scroll'
110
+ private def set(x=nil, y=nil)
111
+ `#@scrolling_native.scrollTop = #{y}` if y
112
+ `#@scrolling_native.scrollLeft = #{x}` if x
113
+ end
24
114
 
25
- `#@native.scrollTop = #{y}`
26
- `#@native.scrollLeft = #{x}`
115
+ def position
116
+ Browser::Position.new(`#@scrolling_native.scrollLeft`, `#@scrolling_native.scrollTop`)
117
+ end
118
+ else
119
+ private def set(x=nil, y=nil)
120
+ raise NotImplementedError, 'scroll on element unsupported'
121
+ end
27
122
 
28
- self
123
+ def position
124
+ raise NotImplementedError, 'scroll on element unsupported'
125
+ end
126
+ end
127
+
128
+ # @!attribute [r] x
129
+ # @return [Integer] the scroll position on the x axis
130
+ def x
131
+ position.x
132
+ end
133
+
134
+ # @!attribute [r] y
135
+ # @return [Integer] the scroll position on the y axis
136
+ def y
137
+ position.y
29
138
  end
30
139
 
140
+ # @!attribute [r] height
141
+ # @return [Integer] the height of the scroll
31
142
  def height
32
- `#@native.scrollHeight`
143
+ `#@scrolling_native.scrollHeight`
33
144
  end
34
145
 
146
+ # @!attribute [r] width
147
+ # @return [Integer] the width of the scroll
35
148
  def width
36
- `#@native.scrollWidth`
149
+ `#@scrolling_native.scrollWidth`
37
150
  end
38
151
 
39
- def by(what)
40
- x = what[:x] || 0
41
- y = what[:y] || 0
42
-
43
- `#@native.scrollBy(#{x}, #{y})`
152
+ if Browser.supports? 'Element.scrollIntoViewIfNeeded'
153
+ def into_view(align = true)
154
+ `#@scrolling_native.scrollIntoViewIfNeeded(align)`
155
+ end
156
+ else
157
+ # Non-standard. Not supported by modern Firefox. Use {#into_view!}
158
+ def into_view(align = true)
159
+ raise NotImplementedError
160
+ end
161
+ end
44
162
 
45
- self
163
+ def into_view!(align = true)
164
+ `#@scrolling_native.scrollIntoView(align)`
46
165
  end
47
166
  end
48
167
 
@@ -0,0 +1,42 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Select < Element
4
+ def_selector "select"
5
+
6
+ def value
7
+ %x{
8
+ if (#@native.value == "") {
9
+ return nil;
10
+ }
11
+ else {
12
+ return #@native.value;
13
+ }
14
+ }
15
+ end
16
+
17
+ def value= value
18
+ `#@native.value = #{value.to_n}`
19
+ end
20
+
21
+ def labels
22
+ NodeSet[Native::Array.new(`#@native.labels`)]
23
+ end
24
+
25
+ def options
26
+ NodeSet[Native::Array.new(`#@native.options`)]
27
+ end
28
+
29
+ def option
30
+ DOM(`#@native.options[#@native.selectedIndex]`)
31
+ end
32
+
33
+ def index
34
+ `#@native.selectedIndex`
35
+ end
36
+
37
+ alias_native :multiple?, :multiple
38
+ alias_native :required?, :required
39
+ alias_native :length
40
+ end
41
+
42
+ end; end; end
@@ -0,0 +1,46 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Size
4
+ attr_reader :element
5
+
6
+ # @private
7
+ def initialize(element, *inc)
8
+ @element = element
9
+ @native = element.to_n
10
+ @include = inc
11
+ end
12
+
13
+ # @!attribute width
14
+ # @return [Integer] the element width
15
+ def width
16
+ `#@native.offsetWidth`
17
+ end
18
+
19
+ def width=(value)
20
+ @element.style[:width] = value
21
+ end
22
+
23
+ # @!attribute height
24
+ # @return [Integer] the element height
25
+ def height
26
+ `#@native.offsetHeight`
27
+ end
28
+
29
+ def height=(value)
30
+ @element.style[:height] = value
31
+ end
32
+
33
+ # @!attribute client_width
34
+ # @return [Integer] the content-box width of an element
35
+ def client_width
36
+ `#@native.clientWidth`
37
+ end
38
+
39
+ # @!attribute client_height
40
+ # @return [Integer] the content-box height of an element
41
+ def client_height
42
+ `#@native.clientHeight`
43
+ end
44
+ end
45
+
46
+ end; end; end
@@ -0,0 +1,11 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Template < Element
4
+ def_selector "template"
5
+
6
+ def content
7
+ DOM(`#@native.content`)
8
+ end
9
+ end
10
+
11
+ end; end; end
@@ -0,0 +1,26 @@
1
+ module Browser; module DOM; class Element < Node
2
+
3
+ class Textarea < Element
4
+ def_selector "textarea"
5
+
6
+ def value
7
+ %x{
8
+ if (#@native.value == "") {
9
+ return nil;
10
+ }
11
+ else {
12
+ return #@native.value;
13
+ }
14
+ }
15
+ end
16
+
17
+ def value=(value)
18
+ `#@native.value = #{value}`
19
+ end
20
+
21
+ def clear
22
+ `#@native.value = ''`
23
+ end
24
+ end
25
+
26
+ end; end; end