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,94 @@
1
+ module Browser
2
+
3
+ class Blob
4
+ include NativeCachedWrapper
5
+
6
+ # Create a new blob from anything that Blob API supports
7
+ def self.create(from, options={})
8
+ new(`new Blob(#{Native.convert(from)}, #{options.to_n})`)
9
+ end
10
+
11
+ # @!attribute [r] size
12
+ # @return [Integer] blob size in bytes
13
+ def size
14
+ `#@native.size`
15
+ end
16
+
17
+ # @!attribute [r] type
18
+ # @return [String] blob mime type
19
+ def type
20
+ `#@native.type`
21
+ end
22
+
23
+ # Convert a blob to a UTF-8 encoded string.
24
+ #
25
+ # If block is given it will be called with a parameter once we receive
26
+ # the text. Otherwise return a {Promise} which will resolve once we
27
+ # receive it.
28
+ def text(&block)
29
+ promise = nil
30
+ unless block_given?
31
+ promise = Promise.new
32
+ block = proc { |i| promise.resolve(i) }
33
+ end
34
+ `#@native.text().then(#{block.to_n})`
35
+ promise
36
+ end
37
+
38
+ # {Buffer} view into the blob
39
+ #
40
+ # If block is given it will be called with a parameter once we receive
41
+ # the buffer. Otherwise return a {Promise} which will resolve once we
42
+ # receive it.
43
+ def buffer
44
+ promise = nil
45
+ unless block_given?
46
+ promise = Promise.new
47
+ block = proc { |i| promise.resolve(i) }
48
+ end
49
+ resblock = proc { |i| block.call(Buffer.new(i)) }
50
+ `#@native.arrayBuffer().then(#{resblock.to_n})`
51
+ promise
52
+ end
53
+
54
+ # Create a new blob by slicing this blob
55
+ def slice(start, finish=nil)
56
+ Blob.new(`#@native.slice(#{start}, #{finish})`)
57
+ end
58
+
59
+ # Convert a blob to an URL that can be used to reference this blob in DOM
60
+ # eg. display some multimedia
61
+ def to_url(window=$window)
62
+ `#{window.to_n}.URL.createObjectURL(#@native)`
63
+ end
64
+
65
+ # Rename a blob and return a {File} with a new name.
66
+ #
67
+ # @return [File] a renamed blob
68
+ def rename(new_filename)
69
+ File.create([self], new_filename, type: type,
70
+ lastModified: respond_to?(:last_modified) ?
71
+ last_modified : Time.now)
72
+ end
73
+ end
74
+
75
+ class File < Blob
76
+ # Create a new file from anything that File API supports
77
+ def self.create(from, name, options={})
78
+ new(`new File(#{Native.convert(from)}, #{name}, #{options.to_n})`)
79
+ end
80
+
81
+ # @!attribute [r] last_modified
82
+ # @return [Time] last modified date of this file
83
+ def last_modified
84
+ Time.at(`#@native.lastModified`/1000.0)
85
+ end
86
+
87
+ # @!attribute [r] name
88
+ # @return [String] filename
89
+ def name
90
+ `#@native.name`
91
+ end
92
+ end
93
+
94
+ end
@@ -1,13 +1,3 @@
1
- #--
2
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
3
- # Version 2, December 2004
4
- #
5
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
6
- # TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
7
- #
8
- # 0. You just DO WHAT THE FUCK YOU WANT TO.
9
- #++
10
-
11
1
  module Browser; class Canvas
12
2
 
13
3
  class Data
@@ -27,7 +17,7 @@ class Data
27
17
  data
28
18
  end
29
19
 
30
- include Native
20
+ include Native::Wrapper
31
21
 
32
22
  attr_reader :x, :y, :width, :height
33
23
 
@@ -42,7 +32,7 @@ class Data
42
32
  end
43
33
 
44
34
  def length
45
- `#@native.length`
35
+ `#@native.data.length`
46
36
  end
47
37
 
48
38
  def [](index)
@@ -1,17 +1,7 @@
1
- #--
2
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
3
- # Version 2, December 2004
4
- #
5
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
6
- # TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
7
- #
8
- # 0. You just DO WHAT THE FUCK YOU WANT TO.
9
- #++
10
-
11
1
  module Browser; class Canvas
12
2
 
13
3
  class Gradient
14
- include Native
4
+ include Native::Wrapper
15
5
 
16
6
  attr_reader :context
17
7
 
@@ -1,17 +1,7 @@
1
- #--
2
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
3
- # Version 2, December 2004
4
- #
5
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
6
- # TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
7
- #
8
- # 0. You just DO WHAT THE FUCK YOU WANT TO.
9
- #++
10
-
11
1
  module Browser; class Canvas
12
2
 
13
3
  class StyleObject
14
- include Native
4
+ include Native::Wrapper
15
5
 
16
6
  attr_reader :context
17
7
 
@@ -66,10 +56,12 @@ class Style < StyleObject
66
56
 
67
57
  def smooth!
68
58
  `#@native.mozImageSmoothingEnabled = #{@smooth = true}`
59
+ `#@native.imageSmoothingEnabled = #{@smooth = true}`
69
60
  end
70
61
 
71
62
  def no_smooth!
72
63
  `#@native.mozImageSmoothingEnabled = #{@smooth = false}`
64
+ `#@native.imageSmoothingEnabled = #{@smooth = false}`
73
65
  end
74
66
  end
75
67
 
@@ -1,17 +1,7 @@
1
- #--
2
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
3
- # Version 2, December 2004
4
- #
5
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
6
- # TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
7
- #
8
- # 0. You just DO WHAT THE FUCK YOU WANT TO.
9
- #++
10
-
11
1
  module Browser; class Canvas
12
2
 
13
3
  class Text
14
- include Native
4
+ include Native::Wrapper
15
5
 
16
6
  attr_reader :context
17
7
 
@@ -1,13 +1,3 @@
1
- #--
2
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
3
- # Version 2, December 2004
4
- #
5
- # DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE
6
- # TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
7
- #
8
- # 0. You just DO WHAT THE FUCK YOU WANT TO.
9
- #++
10
-
11
1
  require 'browser/canvas/style'
12
2
  require 'browser/canvas/text'
13
3
  require 'browser/canvas/data'
@@ -16,17 +6,33 @@ require 'browser/canvas/gradient'
16
6
  module Browser
17
7
 
18
8
  class Canvas
19
- include Native
9
+ include Native::Wrapper
20
10
 
21
11
  attr_reader :element, :style, :text
22
12
 
23
13
  def initialize(*args)
24
- if DOM::Node === args.first
25
- @element = args.shift
26
- else
14
+ if DOM::Element === args.first
15
+ element = args.shift
16
+
17
+ if DOM::Element::Image === element
18
+ @image = element
19
+ else
20
+ @element = element
21
+ end
22
+ elsif Canvas === args.first
23
+ @image = args.first
24
+ end
25
+
26
+ unless @element
27
27
  @element = $document.create_element('canvas')
28
- @element[:width] = args.shift
29
- @element[:height] = args.shift
28
+
29
+ if @image
30
+ @element[:width] = @image.width
31
+ @element[:height] = @image.height
32
+ else
33
+ @element[:width] = args.shift
34
+ @element[:height] = args.shift
35
+ end
30
36
  end
31
37
 
32
38
  if @element.node_name != 'CANVAS'
@@ -37,6 +43,43 @@ class Canvas
37
43
 
38
44
  @style = Style.new(self)
39
45
  @text = Text.new(self)
46
+
47
+ if @image
48
+ draw_image(@image)
49
+ end
50
+ end
51
+
52
+ def width
53
+ @element[:width].to_i
54
+ end
55
+
56
+ def height
57
+ @element[:height].to_i
58
+ end
59
+
60
+ def width=(new_width)
61
+ @element[:width] = new_width.to_i
62
+ end
63
+
64
+ def height=(new_height)
65
+ @element[:height] = new_height.to_i
66
+ end
67
+
68
+ def append_to(parent)
69
+ @element.append_to(parent)
70
+ end
71
+
72
+ def load(path)
73
+ promise = Promise.new
74
+ image = $document.create_element('img')
75
+
76
+ image.on :load do
77
+ promise.resolve(image)
78
+ end
79
+
80
+ image[:src] = path
81
+
82
+ promise
40
83
  end
41
84
 
42
85
  def data(x = nil, y = nil, width = nil, height = nil)
@@ -49,7 +92,7 @@ class Canvas
49
92
  end
50
93
 
51
94
  def pattern(image, type = :repeat)
52
- `#@native.createPattern(#{Element(image).to_n}, type)`
95
+ `#@native.createPattern(#{DOM(image).to_n}, type)`
53
96
  end
54
97
 
55
98
  def gradient(*args, &block)
@@ -143,17 +186,28 @@ class Canvas
143
186
  self
144
187
  end
145
188
 
146
- def draw_image(*args)
147
- image = Element(args.shift)
189
+ def draw_image(image, *args)
190
+ if Canvas === image
191
+ image = image.element
192
+ else
193
+ image = DOM(image)
194
+ end
148
195
 
149
196
  if args.first.is_a?(Hash)
150
197
  source, destination = args
151
198
 
152
199
  `#@native.drawImage(#{image.to_n}, #{source[:x]}, #{source[:y]}, #{source[:width]}, #{source[:height]}, #{destination[:x]}, #{destination[:y]}, #{destination[:width]}, #{destination[:height]})`
153
200
  else
154
- x, y, width, height = args
201
+ case args.length
202
+ when 0
203
+ `#@native.drawImage(#{image.to_n}, 0, 0)`
204
+
205
+ when 2
206
+ `#@native.drawImage(#{image.to_n}, #{args[0]}, #{args[1]})`
155
207
 
156
- `#@native.drawImage(#{image.to_n}, #{x}, #{y}, #{width || `undefined`}, #{height || `undefined`})`
208
+ when 4
209
+ `#@native.drawImage(#{image.to_n}, #{args[0]}, #{args[1]}, #{args[2]}, #{args[3]})`
210
+ end
157
211
  end
158
212
 
159
213
  self
@@ -261,17 +315,21 @@ class Canvas
261
315
  `#@native.isPointInPath(x, y)`
262
316
  end
263
317
 
264
- def width
265
- @element[:width]
318
+ def to_data(type = undefined)
319
+ `#{@element.to_n}.toDataUrl(type)`
266
320
  end
267
321
 
268
- def height
269
- @element[:height]
322
+ def to_dom(*)
323
+ @element
270
324
  end
271
325
 
272
- def to_data(type = undefined)
273
- `#{@element.to_n}.toDataUrl(type)`
274
- end
326
+ def on(*args, &block); @element.on(*args, &block); end
327
+ def one(*args, &block); @element.one(*args, &block); end
328
+ def off(*args, &block); @element.off(*args, &block); end
329
+ end
330
+
331
+ Browser::DOM::Builder.for Canvas do |b, item|
332
+ item.element
275
333
  end
276
334
 
277
335
  end
@@ -1,67 +1,45 @@
1
+ warn "`console' has been moved to Opal's stdlib, please `require 'console'` instead." if RUBY_ENGINE_VERSION.to_f >= 0.9
2
+
1
3
  module Browser
2
4
 
3
- # This class wraps a `window.console`.
5
+ # Manipulate the browser console.
6
+ #
7
+ # @see https://developer.mozilla.org/en-US/docs/Web/API/console
4
8
  class Console
5
- include Native
9
+ include Browser::NativeCachedWrapper
6
10
 
7
11
  # Clear the console.
8
- #
9
- # @return [self]
10
12
  def clear
11
13
  `#@native.clear()`
12
-
13
- self
14
14
  end
15
15
 
16
16
  # Print a stacktrace from the call site.
17
- #
18
- # @return [self]
19
17
  def trace
20
18
  `#@native.trace()`
21
-
22
- self
23
19
  end
24
20
 
25
21
  # Log the passed objects based on an optional initial format.
26
- #
27
- # @return [self]
28
22
  def log(*args)
29
23
  `#@native.log.apply(#@native, args)`
30
-
31
- self
32
24
  end
33
25
 
34
26
  # Log the passed objects based on an optional initial format as informational
35
27
  # log.
36
- #
37
- # @return [self]
38
28
  def info(*args)
39
29
  `#@native.info.apply(#@native, args)`
40
-
41
- self
42
30
  end
43
31
 
44
32
  # Log the passed objects based on an optional initial format as warning.
45
- #
46
- # @return [self]
47
33
  def warn(*args)
48
34
  `#@native.warn.apply(#@native, args)`
49
-
50
- self
51
35
  end
52
36
 
53
37
  # Log the passed objects based on an optional initial format as error.
54
- #
55
- # @return [self]
56
38
  def error(*args)
57
39
  `#@native.error.apply(#@native, args)`
58
-
59
- self
60
40
  end
61
41
 
62
42
  # Time the given block with the given label.
63
- #
64
- # @return [self]
65
43
  def time(label, &block)
66
44
  raise ArgumentError, "no block given" unless block
67
45
 
@@ -76,13 +54,9 @@ class Console
76
54
  ensure
77
55
  `#@native.timeEnd()`
78
56
  end
79
-
80
- self
81
57
  end
82
58
 
83
59
  # Group the given block.
84
- #
85
- # @return [self]
86
60
  def group(*args, &block)
87
61
  raise ArgumentError, "no block given" unless block
88
62
 
@@ -97,13 +71,9 @@ class Console
97
71
  ensure
98
72
  `#@native.groupEnd()`
99
73
  end
100
-
101
- self
102
74
  end
103
75
 
104
76
  # Group the given block but collapse it.
105
- #
106
- # @return [self]
107
77
  def group!(*args, &block)
108
78
  return unless block_given?
109
79
 
@@ -118,8 +88,6 @@ class Console
118
88
  ensure
119
89
  `#@native.groupEnd()`
120
90
  end
121
-
122
- self
123
91
  end
124
92
  end
125
93
 
@@ -1,78 +1,141 @@
1
+ require 'stringio'
2
+
1
3
  module Browser
2
4
 
5
+ # Allows manipulation of browser cookies.
6
+ #
7
+ # @see https://developer.mozilla.org/en-US/docs/Web/API/document.cookie
8
+ #
9
+ # Usage:
10
+ #
11
+ # cookies = Browser::Cookies.new(`document`)
12
+ # cookies["my-cookie"] = "monster"
13
+ # cookies.delete("my-cookie")
14
+ #
3
15
  class Cookies
16
+ # Default cookie options.
17
+ DEFAULT = {
18
+ expires: Time.now + 60 * 60 * 24,
19
+ secure: false
20
+ }
21
+
4
22
  include Enumerable
5
23
 
24
+ attr_reader :options
25
+
26
+ # Create a new {Cookies} wrapper.
27
+ #
28
+ # @param document [native] the native document object
6
29
  def initialize(document)
7
30
  @document = document
8
-
9
- @options = {
10
- expires: Time.now + 1.day,
11
- path: '',
12
- domain: '',
13
- secure: ''
14
- }
15
- end
16
-
17
- def options (value = nil)
18
- value ? @options.merge!(value) : @options
31
+ @options = DEFAULT.dup
19
32
  end
20
33
 
21
- def [] (name)
22
- matches = `#@document.cookie`.scan(/#{Regexp.escape(key.encode_uri_component)}=([^;]*)/)
34
+ # Access the cookie with the given name.
35
+ #
36
+ # @param name [String] the name of the cookie
37
+ #
38
+ # @return [Object]
39
+ def [](name)
40
+ matches = `#@document.cookie`.scan(/#{Regexp.escape(FormData.encode(name))}=([^;]*)/)
23
41
 
24
42
  return if matches.empty?
25
43
 
26
- result = matches.map {|cookie|
27
- JSON.parse(cookie.match(/^.*?=(.*)$/)[1].decode_uri_component)
44
+ result = matches.flatten.map {|value|
45
+ JSON.parse(FormData.decode(value))
28
46
  }
29
47
 
30
48
  result.length == 1 ? result.first : result
31
49
  end
32
50
 
33
- def []= (name, value, options = {})
34
- `#@document.cookie = #{encode name, value.is_a?(String) ? value : JSON.dump(value), @options.merge(options)}`
51
+ # Set a cookie.
52
+ #
53
+ # @param name [String] the name of the cookie
54
+ # @param value [Object] the data to set
55
+ # @param options [Hash] the options for the cookie
56
+ #
57
+ # @option options [Integer] :max_age the max age of the cookie in seconds
58
+ # @option options [Time] :expires the expire date
59
+ # @option options [String] :path the path the cookie is valid on
60
+ # @option options [String] :domain the domain the cookie is valid on
61
+ # @option options [Boolean] :secure whether the cookie is secure or not
62
+ def []=(name, value, options = {})
63
+ string = value.is_a?(String) ? value : JSON.dump(value)
64
+ encoded_value = encode(name, string, @options.merge(options))
65
+ `#@document.cookie = #{encoded_value}`
35
66
  end
36
67
 
37
- def delete (name)
68
+ # Delete a cookie.
69
+ #
70
+ # @param name [String] the name of the cookie
71
+ def delete(name)
38
72
  `#@document.cookie = #{encode name, '', expires: Time.now}`
39
73
  end
40
74
 
75
+ # @!attribute [r] keys
76
+ # @return [Array<String>] all the cookie names
41
77
  def keys
42
78
  Array(`#@document.cookie.split(/; /)`).map {|cookie|
43
79
  cookie.split(/\s*=\s*/).first
44
80
  }
45
81
  end
46
82
 
83
+ # @!attribute [r] values
84
+ # @return [Array] all the cookie values
47
85
  def values
48
86
  keys.map {|key|
49
87
  self[key]
50
88
  }
51
89
  end
52
90
 
53
- def each
91
+ # Enumerate the cookies.
92
+ #
93
+ # @yieldparam key [String] the name of the cookie
94
+ # @yieldparam value [String] the value of the cookie
95
+ #
96
+ # @return [self]
97
+ def each(&block)
98
+ return enum_for :each unless block
99
+
54
100
  keys.each {|key|
55
101
  yield key, self[key]
56
102
  }
103
+
104
+ self
57
105
  end
58
106
 
107
+ # Delete all the cookies
108
+ #
109
+ # @return [self]
59
110
  def clear
60
111
  keys.each {|key|
61
112
  delete key
62
113
  }
114
+
115
+ self
63
116
  end
64
117
 
65
118
  protected
66
- def encode (key, value, options = {})
67
- result = "#{key.encode_uri_component}=#{value.encode_uri_component}; "
119
+ def encode(key, value, options = {})
120
+ io = StringIO.new
121
+
122
+ io << FormData.encode(key) << ?= << FormData.encode(value) << '; '
68
123
 
69
- result += "max-age=#{options[:max_age]}; " if options[:max_age]
70
- result += "expires=#{options[:expires].to_utc}; " if options[:expires]
71
- result += "path=#{options[:path]}; " if options[:path]
72
- result += "domain=#{options[:domain]}; " if options[:domain]
73
- result += 'secure' if options[:secure]
124
+ io << 'max-age=' << options[:max_age] << '; ' if options[:max_age]
125
+ io << 'expires=' << options[:expires].utc << '; ' if options[:expires]
126
+ io << 'path=' << options[:path] << '; ' if options[:path]
127
+ io << 'domain=' << options[:domain] << '; ' if options[:domain]
128
+ io << 'secure' if options[:secure]
129
+
130
+ io.string
131
+ end
132
+ end
74
133
 
75
- result
134
+ class DOM::Document < DOM::Element
135
+ # @!attribute [r] cookies
136
+ # @return [Cookies] the cookies for the document
137
+ def cookies
138
+ Cookies.new(@native) if defined?(`#@native.cookie`)
76
139
  end
77
140
  end
78
141