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.
- checksums.yaml +5 -5
- data/.github/workflows/build.yml +95 -0
- data/.gitignore +3 -0
- data/.yardopts +1 -1
- data/Gemfile +22 -3
- data/LICENSE +20 -0
- data/README.md +200 -20
- data/Rakefile +29 -1
- data/config.ru +20 -2
- data/docs/polyfills.md +24 -0
- data/examples/2048/Gemfile +6 -0
- data/examples/2048/README.md +13 -0
- data/examples/2048/app/application.rb +169 -0
- data/examples/2048/config.ru +9 -0
- data/examples/canvas/Gemfile +6 -0
- data/examples/canvas/README.md +9 -0
- data/examples/canvas/app/application.rb +55 -0
- data/examples/canvas/config.ru +9 -0
- data/examples/component/Gemfile +6 -0
- data/examples/component/README.md +10 -0
- data/examples/component/app/application.rb +66 -0
- data/examples/component/config.ru +9 -0
- data/examples/integrations/README.md +24 -0
- data/examples/integrations/dynamic-rack-opal-sprockets-server/Gemfile +6 -0
- data/examples/integrations/dynamic-rack-opal-sprockets-server/README.md +16 -0
- data/examples/integrations/dynamic-rack-opal-sprockets-server/app/application.rb +6 -0
- data/examples/integrations/dynamic-rack-opal-sprockets-server/config.ru +9 -0
- data/examples/integrations/dynamic-roda-roda-sprockets/.gitignore +1 -0
- data/examples/integrations/dynamic-roda-roda-sprockets/Gemfile +7 -0
- data/examples/integrations/dynamic-roda-roda-sprockets/README.md +22 -0
- data/examples/integrations/dynamic-roda-roda-sprockets/Rakefile +4 -0
- data/examples/integrations/dynamic-roda-roda-sprockets/app/application.rb +6 -0
- data/examples/integrations/dynamic-roda-roda-sprockets/app.rb +32 -0
- data/examples/integrations/dynamic-roda-roda-sprockets/config.ru +3 -0
- data/examples/integrations/dynamic-roda-tilt/.gitignore +1 -0
- data/examples/integrations/dynamic-roda-tilt/Gemfile +8 -0
- data/examples/integrations/dynamic-roda-tilt/README.md +17 -0
- data/examples/integrations/dynamic-roda-tilt/Rakefile +6 -0
- data/examples/integrations/dynamic-roda-tilt/app/application.rb +6 -0
- data/examples/integrations/dynamic-roda-tilt/app.rb +50 -0
- data/examples/integrations/dynamic-roda-tilt/config.ru +3 -0
- data/examples/integrations/dynamic-sinatra-opal-sprockets-server/Gemfile +7 -0
- data/examples/integrations/dynamic-sinatra-opal-sprockets-server/README.md +16 -0
- data/examples/integrations/dynamic-sinatra-opal-sprockets-server/app/application.rb +6 -0
- data/examples/integrations/dynamic-sinatra-opal-sprockets-server/config.ru +29 -0
- data/examples/integrations/static-bash/.gitignore +2 -0
- data/examples/integrations/static-bash/Gemfile +3 -0
- data/examples/integrations/static-bash/README.md +8 -0
- data/examples/integrations/static-bash/app/application.rb +6 -0
- data/examples/integrations/static-bash/build.sh +4 -0
- data/examples/integrations/static-bash/index.html +10 -0
- data/examples/integrations/static-bash-opal-parser/.gitignore +3 -0
- data/examples/integrations/static-bash-opal-parser/Gemfile +3 -0
- data/examples/integrations/static-bash-opal-parser/README.md +10 -0
- data/examples/integrations/static-bash-opal-parser/build.sh +4 -0
- data/examples/integrations/static-bash-opal-parser/index.html +19 -0
- data/examples/integrations/static-rake/.gitignore +1 -0
- data/examples/integrations/static-rake/Gemfile +4 -0
- data/examples/integrations/static-rake/README.md +7 -0
- data/examples/integrations/static-rake/Rakefile +10 -0
- data/examples/integrations/static-rake/app/application.rb +6 -0
- data/examples/integrations/static-rake/index.html +9 -0
- data/examples/integrations/static-rake-guard/.gitignore +1 -0
- data/examples/integrations/static-rake-guard/Gemfile +6 -0
- data/examples/integrations/static-rake-guard/Guardfile +3 -0
- data/examples/integrations/static-rake-guard/README.md +10 -0
- data/examples/integrations/static-rake-guard/Rakefile +10 -0
- data/examples/integrations/static-rake-guard/app/application.rb +6 -0
- data/examples/integrations/static-rake-guard/index.html +9 -0
- data/examples/svg/.gitignore +1 -0
- data/examples/svg/Gemfile +4 -0
- data/examples/svg/README.md +7 -0
- data/examples/svg/Rakefile +10 -0
- data/examples/svg/app/application.rb +11 -0
- data/examples/svg/index.html +17 -0
- data/examples/svg/index.svg +6 -0
- data/index.html.erb +24 -0
- data/lib/opal-browser.rb +1 -0
- data/opal/browser/animation_frame.rb +92 -10
- data/opal/browser/audio/node.rb +121 -0
- data/opal/browser/audio/param_schedule.rb +43 -0
- data/opal/browser/audio.rb +66 -0
- data/opal/browser/blob.rb +94 -0
- data/opal/browser/canvas/data.rb +2 -12
- data/opal/browser/canvas/gradient.rb +1 -11
- data/opal/browser/canvas/style.rb +3 -11
- data/opal/browser/canvas/text.rb +1 -11
- data/opal/browser/canvas.rb +86 -28
- data/opal/browser/console.rb +6 -38
- data/opal/browser/cookies.rb +90 -27
- data/opal/browser/crypto.rb +79 -0
- data/opal/browser/css/declaration.rb +1 -6
- data/opal/browser/css/rule.rb +1 -1
- data/opal/browser/css/style_sheet.rb +2 -2
- data/opal/browser/css.rb +23 -7
- data/opal/browser/database/sql.rb +193 -0
- data/opal/browser/delay.rb +94 -0
- data/opal/browser/dom/attribute.rb +16 -9
- data/opal/browser/dom/builder.rb +35 -25
- data/opal/browser/dom/character_data.rb +43 -7
- data/opal/browser/dom/document.rb +171 -37
- data/opal/browser/dom/document_fragment.rb +18 -0
- data/opal/browser/dom/document_or_shadow_root.rb +19 -0
- data/opal/browser/dom/element/attributes.rb +111 -0
- data/opal/browser/dom/element/button.rb +31 -0
- data/opal/browser/dom/element/custom.rb +177 -0
- data/opal/browser/dom/element/data.rb +82 -0
- data/opal/browser/dom/element/editable.rb +47 -0
- data/opal/browser/dom/element/form.rb +38 -0
- data/opal/browser/dom/element/iframe.rb +37 -0
- data/opal/browser/dom/element/image.rb +25 -0
- data/opal/browser/dom/element/input.rb +48 -1
- data/opal/browser/dom/element/media.rb +17 -0
- data/opal/browser/dom/element/offset.rb +32 -10
- data/opal/browser/dom/element/position.rb +11 -2
- data/opal/browser/dom/element/scroll.rb +139 -20
- data/opal/browser/dom/element/select.rb +42 -0
- data/opal/browser/dom/element/size.rb +46 -0
- data/opal/browser/dom/element/template.rb +11 -0
- data/opal/browser/dom/element/textarea.rb +26 -0
- data/opal/browser/dom/element.rb +496 -168
- data/opal/browser/dom/mutation_observer.rb +69 -9
- data/opal/browser/dom/node.rb +270 -83
- data/opal/browser/dom/node_set.rb +74 -41
- data/opal/browser/dom/shadow_root.rb +12 -0
- data/opal/browser/dom/text.rb +18 -3
- data/opal/browser/dom.rb +40 -18
- data/opal/browser/effects.rb +180 -3
- data/opal/browser/event/all.rb +26 -0
- data/opal/browser/event/animation.rb +40 -0
- data/opal/browser/{dom/event → event}/audio_processing.rb +10 -6
- data/opal/browser/event/base.rb +461 -0
- data/opal/browser/event/before_unload.rb +17 -0
- data/opal/browser/event/clipboard.rb +37 -0
- data/opal/browser/event/close.rb +49 -0
- data/opal/browser/event/composition.rb +52 -0
- data/opal/browser/event/custom.rb +65 -0
- data/opal/browser/event/data_transfer.rb +95 -0
- data/opal/browser/event/device_light.rb +25 -0
- data/opal/browser/{dom/event → event}/device_motion.rb +21 -6
- data/opal/browser/event/device_orientation.rb +50 -0
- data/opal/browser/{dom/event → event}/device_proximity.rb +10 -6
- data/opal/browser/event/drag.rb +123 -0
- data/opal/browser/event/focus.rb +41 -0
- data/opal/browser/event/gamepad.rb +62 -0
- data/opal/browser/{dom/event → event}/hash_change.rb +10 -6
- data/opal/browser/event/keyboard.rb +128 -0
- data/opal/browser/event/message.rb +72 -0
- data/opal/browser/{dom/event → event}/mouse.rb +37 -32
- data/opal/browser/event/page_transition.rb +25 -0
- data/opal/browser/event/pop_state.rb +35 -0
- data/opal/browser/event/progress.rb +45 -0
- data/opal/browser/event/sensor.rb +17 -0
- data/opal/browser/{dom/event → event}/storage.rb +10 -6
- data/opal/browser/{dom/event → event}/touch.rb +14 -21
- data/opal/browser/event/ui.rb +38 -0
- data/opal/browser/{dom/event → event}/wheel.rb +6 -4
- data/opal/browser/event.rb +163 -0
- data/opal/browser/event_source.rb +7 -4
- data/opal/browser/form_data.rb +225 -0
- data/opal/browser/history.rb +53 -21
- data/opal/browser/http/binary.rb +1 -0
- data/opal/browser/http/headers.rb +21 -2
- data/opal/browser/http/request.rb +83 -55
- data/opal/browser/http/response.rb +5 -1
- data/opal/browser/http.rb +47 -9
- data/opal/browser/immediate.rb +128 -10
- data/opal/browser/interval.rb +41 -23
- data/opal/browser/location.rb +20 -4
- data/opal/browser/navigator.rb +136 -13
- data/opal/browser/polyfill/visual_viewport.rb +216 -0
- data/opal/browser/screen.rb +34 -8
- data/opal/browser/setup/base.rb +6 -0
- data/opal/browser/setup/full.rb +13 -0
- data/opal/browser/setup/large.rb +17 -0
- data/opal/browser/setup/mini.rb +8 -0
- data/opal/browser/setup/traditional.rb +10 -0
- data/opal/browser/socket.rb +16 -8
- data/opal/browser/storage.rb +155 -52
- data/opal/browser/support.rb +299 -0
- data/opal/browser/utils.rb +116 -18
- data/opal/browser/version.rb +1 -1
- data/opal/browser/visual_viewport.rb +39 -0
- data/opal/browser/window/size.rb +47 -9
- data/opal/browser/window/view.rb +37 -4
- data/opal/browser/window.rb +46 -26
- data/opal/browser.rb +1 -10
- data/opal/opal-browser.rb +1 -0
- data/opal-browser.gemspec +10 -12
- data/spec/database/sql_spec.rb +139 -0
- data/spec/delay_spec.rb +41 -0
- data/spec/dom/attribute_spec.rb +49 -0
- data/spec/dom/builder_spec.rb +36 -19
- data/spec/dom/document_spec.rb +28 -6
- data/spec/dom/element/attributes_spec.rb +52 -0
- data/spec/dom/element/custom_spec.rb +106 -0
- data/spec/dom/element/subclass_spec.rb +144 -0
- data/spec/dom/element_spec.rb +184 -7
- data/spec/dom/mutation_observer_spec.rb +13 -9
- data/spec/dom/node_set_spec.rb +44 -0
- data/spec/dom/node_spec.rb +87 -27
- data/spec/dom_spec.rb +19 -9
- data/spec/event_source_spec.rb +18 -15
- data/spec/{dom/event_spec.rb → event_spec.rb} +55 -26
- data/spec/history_spec.rb +32 -19
- data/spec/http_spec.rb +25 -36
- data/spec/immediate_spec.rb +10 -7
- data/spec/interval_spec.rb +59 -0
- data/spec/native_cached_wrapper_spec.rb +46 -0
- data/spec/runner.rb +107 -0
- data/spec/socket_spec.rb +18 -14
- data/spec/spec_helper.rb +2 -4
- data/spec/spec_helper_promise.rb.erb +25 -0
- data/spec/storage_spec.rb +7 -7
- data/spec/wgxpath.install.js +49 -0
- data/spec/window_spec.rb +2 -2
- metadata +181 -93
- data/opal/browser/compatibility/animation_frame.rb +0 -93
- data/opal/browser/compatibility/dom/document/window.rb +0 -15
- data/opal/browser/compatibility/dom/element/css.rb +0 -15
- data/opal/browser/compatibility/dom/element/matches.rb +0 -31
- data/opal/browser/compatibility/dom/element/offset.rb +0 -20
- data/opal/browser/compatibility/dom/element/scroll.rb +0 -25
- data/opal/browser/compatibility/dom/element/style.rb +0 -15
- data/opal/browser/compatibility/dom/mutation_observer.rb +0 -47
- data/opal/browser/compatibility/http/request.rb +0 -15
- data/opal/browser/compatibility/immediate.rb +0 -107
- data/opal/browser/compatibility/window/scroll.rb +0 -27
- data/opal/browser/compatibility/window/size.rb +0 -13
- data/opal/browser/compatibility/window/view.rb +0 -13
- data/opal/browser/compatibility.rb +0 -59
- data/opal/browser/dom/compatibility.rb +0 -8
- data/opal/browser/dom/event/animation.rb +0 -26
- data/opal/browser/dom/event/base.rb +0 -207
- data/opal/browser/dom/event/before_unload.rb +0 -13
- data/opal/browser/dom/event/clipboard.rb +0 -26
- data/opal/browser/dom/event/close.rb +0 -35
- data/opal/browser/dom/event/composition.rb +0 -38
- data/opal/browser/dom/event/custom.rb +0 -30
- data/opal/browser/dom/event/device_light.rb +0 -21
- data/opal/browser/dom/event/device_orientation.rb +0 -36
- data/opal/browser/dom/event/drag.rb +0 -113
- data/opal/browser/dom/event/focus.rb +0 -23
- data/opal/browser/dom/event/gamepad.rb +0 -47
- data/opal/browser/dom/event/keyboard.rb +0 -93
- data/opal/browser/dom/event/message.rb +0 -50
- data/opal/browser/dom/event/page_transition.rb +0 -21
- data/opal/browser/dom/event/pop_state.rb +0 -21
- data/opal/browser/dom/event/progress.rb +0 -31
- data/opal/browser/dom/event/sensor.rb +0 -13
- data/opal/browser/dom/event/ui.rb +0 -22
- data/opal/browser/dom/event.rb +0 -240
- data/opal/browser/http/compatibility.rb +0 -1
- data/opal/browser/http/parameters.rb +0 -8
- data/opal/browser/timeout.rb +0 -60
- data/opal/browser/window/compatibility.rb +0 -3
- data/opal/browser/window/scroll.rb +0 -49
@@ -0,0 +1,79 @@
|
|
1
|
+
module Browser
|
2
|
+
|
3
|
+
# Implements (parts of) the web crypto interface
|
4
|
+
#
|
5
|
+
# https://developer.mozilla.org/en-US/docs/Web/API/Web_Crypto_API
|
6
|
+
class Crypto
|
7
|
+
include NativeCachedWrapper
|
8
|
+
|
9
|
+
class Digest
|
10
|
+
def initialize(buf)
|
11
|
+
@buffer = Buffer.new(buf)
|
12
|
+
end
|
13
|
+
|
14
|
+
attr_reader :buffer
|
15
|
+
|
16
|
+
# Convert a digest to a hexadecimal string
|
17
|
+
def to_hex
|
18
|
+
buffer.to_a.map { |i| "%02x" % i }.join
|
19
|
+
end
|
20
|
+
|
21
|
+
# Convert a digest to a binary string
|
22
|
+
def to_s
|
23
|
+
buffer.to_a.map { |i| "%c" % i }.join
|
24
|
+
end
|
25
|
+
|
26
|
+
# Convert a digest to a Base64-encoded string
|
27
|
+
#
|
28
|
+
# You will need to `require "base64"`
|
29
|
+
def to_b64
|
30
|
+
Base64.strict_encode64(to_s)
|
31
|
+
end
|
32
|
+
|
33
|
+
# Convert a digest to a urlsafe Base64-encoded string
|
34
|
+
#
|
35
|
+
# You will need to `require "base64"`
|
36
|
+
def to_u64(padding: false)
|
37
|
+
Base64.urlsafe_encode64(to_s, padding: padding)
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
# Compute a cryptographic digest of data (a Buffer). If block is given,
|
42
|
+
# it will call a block, otherwise it will return a Promise that will
|
43
|
+
# return once a digest is computed.
|
44
|
+
#
|
45
|
+
# Allowed values for algo: SHA-1, SHA-256 (default), SHA-368, SHA-512.
|
46
|
+
#
|
47
|
+
# The block/promise will be given an argument of type {Digest} which can
|
48
|
+
# be used to format a digest.
|
49
|
+
#
|
50
|
+
# Example:
|
51
|
+
# ```
|
52
|
+
# Browser::Blob.new(['test']).buffer.then { |b|
|
53
|
+
# $window.crypto.digest(b)
|
54
|
+
# }.then { |d|
|
55
|
+
# puts d.to_hex
|
56
|
+
# }
|
57
|
+
# ```
|
58
|
+
def digest data, algo = 'SHA-256', &block
|
59
|
+
promise = nil
|
60
|
+
unless block_given?
|
61
|
+
promise = Promise.new
|
62
|
+
block = proc { |i| promise.resolve(i) }
|
63
|
+
end
|
64
|
+
resblock = proc { |i| block.call(Digest.new(i)) }
|
65
|
+
|
66
|
+
`#@native.subtle.digest(algo, #{Native.convert(data)}).then(#{resblock.to_n})`
|
67
|
+
promise
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
class Window
|
72
|
+
# @!attribute [r] crypto
|
73
|
+
# @return [Crypto] the crypto interface of this window
|
74
|
+
def crypto
|
75
|
+
@crypto ||= Crypto.new(`#@native.crypto`)
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
end
|
@@ -1,7 +1,7 @@
|
|
1
1
|
module Browser; module CSS
|
2
2
|
|
3
3
|
class Declaration
|
4
|
-
include
|
4
|
+
include Browser::NativeCachedWrapper
|
5
5
|
include Enumerable
|
6
6
|
|
7
7
|
def rule
|
@@ -18,21 +18,16 @@ class Declaration
|
|
18
18
|
|
19
19
|
def replace(string)
|
20
20
|
`#@native.cssText = #{string}`
|
21
|
-
|
22
|
-
self
|
23
21
|
end
|
24
22
|
|
25
23
|
def apply(&block)
|
26
24
|
Paggio::CSS::Definition.new(&block).each {|style|
|
27
|
-
# FIXME: use ternary operator when it's fixed
|
28
25
|
if style.important
|
29
26
|
`#@native.setProperty(#{style.name}, #{style.value}, "important")`
|
30
27
|
else
|
31
28
|
`#@native.setProperty(#{style.name}, #{style.value}, "")`
|
32
29
|
end
|
33
30
|
}
|
34
|
-
|
35
|
-
self
|
36
31
|
end
|
37
32
|
|
38
33
|
def delete(name)
|
data/opal/browser/css/rule.rb
CHANGED
data/opal/browser/css.rb
CHANGED
@@ -4,19 +4,35 @@ require 'browser/css/rule'
|
|
4
4
|
require 'browser/css/rule/style'
|
5
5
|
|
6
6
|
module Kernel
|
7
|
-
#
|
8
|
-
# {Browser::CSS::Builder} DSL.
|
7
|
+
# @overload CSS(document = $document, &block)
|
9
8
|
#
|
10
|
-
#
|
11
|
-
#
|
12
|
-
|
13
|
-
|
9
|
+
# Create a `<style>` element from a {Paggio::CSS} DSL.
|
10
|
+
#
|
11
|
+
# @param document [Browser::DOM::Document] the document instance
|
12
|
+
# we intend to use
|
13
|
+
#
|
14
|
+
# @return [Browser::DOM::Element] the created `<style>` element
|
15
|
+
#
|
16
|
+
# @overload CSS(string, document = $document)
|
17
|
+
#
|
18
|
+
# Create a `<style>` element from a string.
|
19
|
+
#
|
20
|
+
# @param document [Browser::DOM::Document] the document instance
|
21
|
+
# we intend to use
|
22
|
+
#
|
23
|
+
# @return [Browser::DOM::Element] the created `<style>` element
|
24
|
+
def CSS(*args, &block)
|
25
|
+
document = if args.length > 1 || block_given?
|
26
|
+
args.pop
|
27
|
+
end || $document
|
28
|
+
|
29
|
+
style = document.create_element(:style)
|
14
30
|
style[:type] = 'text/css'
|
15
31
|
|
16
32
|
if block
|
17
33
|
style.inner_text = Paggio.css(&block)
|
18
34
|
else
|
19
|
-
style.inner_text =
|
35
|
+
style.inner_text = args.join("")
|
20
36
|
end
|
21
37
|
|
22
38
|
style
|
@@ -0,0 +1,193 @@
|
|
1
|
+
require 'ostruct'
|
2
|
+
|
3
|
+
module Browser; module Database
|
4
|
+
|
5
|
+
class SQL
|
6
|
+
# Check if the browser supports WebSQL.
|
7
|
+
def self.supported?
|
8
|
+
Browser.supports? 'WebSQL'
|
9
|
+
end
|
10
|
+
|
11
|
+
class Error < StandardError
|
12
|
+
def self.new(error)
|
13
|
+
return super if self != Error
|
14
|
+
|
15
|
+
[Unknown, Database, Version, TooLarge, Quota, Syntax, Constraint, Timeout] \
|
16
|
+
[`error.code`].new(`error.message`)
|
17
|
+
end
|
18
|
+
|
19
|
+
Unknown = Class.new(self)
|
20
|
+
Database = Class.new(self)
|
21
|
+
Version = Class.new(self)
|
22
|
+
TooLarge = Class.new(self)
|
23
|
+
Quota = Class.new(self)
|
24
|
+
Syntax = Class.new(self)
|
25
|
+
Constraint = Class.new(self)
|
26
|
+
Timeout = Class.new(self)
|
27
|
+
end
|
28
|
+
|
29
|
+
include Native::Wrapper
|
30
|
+
|
31
|
+
# @return [String] the name of the database
|
32
|
+
attr_reader :name
|
33
|
+
|
34
|
+
# @return [String] the description for the database
|
35
|
+
attr_reader :description
|
36
|
+
|
37
|
+
# @return [Integer] the size constraint in bytes
|
38
|
+
attr_reader :size
|
39
|
+
|
40
|
+
# Open a database with the given name and options.
|
41
|
+
#
|
42
|
+
# @param name [String] the name for the database
|
43
|
+
# @param options [Hash] options to open the database
|
44
|
+
#
|
45
|
+
# @option options [String] :description the description for the database
|
46
|
+
# @option options [String] :version ('') the expected version of the database
|
47
|
+
# @option options [Integer] :size (5 * 1024 * 1024) the size constraint in bytes
|
48
|
+
def initialize(name, options = {})
|
49
|
+
@name = name
|
50
|
+
@description = options[:description] || name
|
51
|
+
@version = options[:version] || ''
|
52
|
+
@size = options[:size] || 2 * 1024 * 1024
|
53
|
+
|
54
|
+
super(`window.openDatabase(#{name}, #{@version}, #{@description}, #{@size})`)
|
55
|
+
end
|
56
|
+
|
57
|
+
# @overload version()
|
58
|
+
#
|
59
|
+
# Get the version of the database.
|
60
|
+
#
|
61
|
+
# @return [String]
|
62
|
+
#
|
63
|
+
# @overload version(from, to, &block)
|
64
|
+
#
|
65
|
+
# Migrate the database to a new version.
|
66
|
+
#
|
67
|
+
# @param from [String] the version you're migrating from
|
68
|
+
# @param to [String] the version you're migrating to
|
69
|
+
#
|
70
|
+
# @yieldparam transaction [Transaction] the transaction to work with
|
71
|
+
def version(from = nil, to = nil, &block)
|
72
|
+
return `#@native.version` unless block
|
73
|
+
|
74
|
+
`#@native.changeVersion(#{from}, #{to},
|
75
|
+
#{->(t) { block.call(Transaction.new(self, t)) }})`
|
76
|
+
end
|
77
|
+
|
78
|
+
# Start a transaction on the database.
|
79
|
+
#
|
80
|
+
# @yieldparam transaction [Transaction] the transaction to work on
|
81
|
+
def transaction(&block)
|
82
|
+
raise ArgumentError, 'no block given' unless block
|
83
|
+
|
84
|
+
`#@native.transaction(#{->(t) { block.call(Transaction.new(self, t)) }})`
|
85
|
+
end
|
86
|
+
|
87
|
+
# Allows you to make changes to the database or read data from it.
|
88
|
+
class Transaction
|
89
|
+
include Native::Wrapper
|
90
|
+
|
91
|
+
# @return [Database] the database the transaction has been created from
|
92
|
+
attr_reader :database
|
93
|
+
|
94
|
+
# @private
|
95
|
+
def initialize(database, transaction)
|
96
|
+
@database = database
|
97
|
+
|
98
|
+
super(transaction)
|
99
|
+
end
|
100
|
+
|
101
|
+
# Query the database.
|
102
|
+
#
|
103
|
+
# @param query [String] the SQL query to send
|
104
|
+
# @param parameters [Array] optional bind parameters for the query
|
105
|
+
#
|
106
|
+
# @return [Promise]
|
107
|
+
def query(query, *parameters)
|
108
|
+
promise = Promise.new
|
109
|
+
|
110
|
+
`#@native.executeSql(#{query}, #{parameters},
|
111
|
+
#{->(_, r) { promise.resolve(Result.new(self, r)) }},
|
112
|
+
#{->(_, e) { promise.reject(Error.new(e)) }})`
|
113
|
+
|
114
|
+
promise
|
115
|
+
end
|
116
|
+
end
|
117
|
+
|
118
|
+
# Represents a row.
|
119
|
+
class Row < OpenStruct
|
120
|
+
# @private
|
121
|
+
def initialize(row)
|
122
|
+
super(Hash.new(row))
|
123
|
+
end
|
124
|
+
|
125
|
+
def inspect
|
126
|
+
"#<SQL::Row: #{Hash.new(@native).inspect}>"
|
127
|
+
end
|
128
|
+
end
|
129
|
+
|
130
|
+
class Result
|
131
|
+
include Native::Wrapper
|
132
|
+
|
133
|
+
# @return [Transaction] the transaction the result came from
|
134
|
+
attr_reader :transaction
|
135
|
+
|
136
|
+
# @return [SQL] the database the result came from
|
137
|
+
attr_reader :database
|
138
|
+
|
139
|
+
# @private
|
140
|
+
def initialize(transaction, result)
|
141
|
+
@transaction = transaction
|
142
|
+
@database = transaction.database
|
143
|
+
|
144
|
+
super(result)
|
145
|
+
end
|
146
|
+
|
147
|
+
include Enumerable
|
148
|
+
|
149
|
+
# Get a row from the result.
|
150
|
+
#
|
151
|
+
# @param index [Integer] the index for the row
|
152
|
+
#
|
153
|
+
# @return [Row]
|
154
|
+
def [](index)
|
155
|
+
if index < 0
|
156
|
+
index += length
|
157
|
+
end
|
158
|
+
|
159
|
+
unless index < 0 || index >= length
|
160
|
+
Row.new(`#@native.rows.item(index)`)
|
161
|
+
end
|
162
|
+
end
|
163
|
+
|
164
|
+
# Enumerate over the rows.
|
165
|
+
#
|
166
|
+
# @yieldparam row [Row]
|
167
|
+
#
|
168
|
+
# @return [self]
|
169
|
+
def each(&block)
|
170
|
+
return enum_for :each unless block
|
171
|
+
|
172
|
+
%x{
|
173
|
+
for (var i = 0, length = #@native.rows.length; i < length; i++) {
|
174
|
+
#{block.call(self[`i`])};
|
175
|
+
}
|
176
|
+
}
|
177
|
+
|
178
|
+
self
|
179
|
+
end
|
180
|
+
|
181
|
+
# @!attribute [r] length
|
182
|
+
# @return [Integer] number of rows in the result
|
183
|
+
def length
|
184
|
+
`#@native.rows.length`
|
185
|
+
end
|
186
|
+
|
187
|
+
# @!attribute [r] affected
|
188
|
+
# @return [Integer] number of affected rows
|
189
|
+
alias_native :affected, :rowsAffected
|
190
|
+
end
|
191
|
+
end
|
192
|
+
|
193
|
+
end; end
|
@@ -0,0 +1,94 @@
|
|
1
|
+
module Browser
|
2
|
+
|
3
|
+
# Allows you to delay the call to a function which gets called after the
|
4
|
+
# given time.
|
5
|
+
#
|
6
|
+
# @see https://developer.mozilla.org/en-US/docs/Web/API/Window.setTimeout
|
7
|
+
class Delay
|
8
|
+
# @!attribute [r] after
|
9
|
+
# @return [Float] the seconds after which the block is called
|
10
|
+
attr_reader :after
|
11
|
+
|
12
|
+
# Create and start a timeout.
|
13
|
+
#
|
14
|
+
# @param window [Window] the window to start the timeout on
|
15
|
+
# @param time [Float] seconds after which the block is called
|
16
|
+
def initialize(window, time, &block)
|
17
|
+
@window = Native.convert(window)
|
18
|
+
@after = time
|
19
|
+
@block = block
|
20
|
+
end
|
21
|
+
|
22
|
+
# Abort the timeout.
|
23
|
+
def abort
|
24
|
+
`#@window.clearTimeout(#@id)`
|
25
|
+
end
|
26
|
+
|
27
|
+
# Start the delay.
|
28
|
+
def start
|
29
|
+
@id = `#@window.setTimeout(#{@block.to_n}, #@after * 1000)`
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
class Window
|
34
|
+
# Execute a block after the given seconds.
|
35
|
+
#
|
36
|
+
# @param time [Float] the seconds after it gets called
|
37
|
+
#
|
38
|
+
# @return [Delay] the object representing the timeout
|
39
|
+
def after(time, &block)
|
40
|
+
Delay.new(@native, time, &block).tap(&:start)
|
41
|
+
end
|
42
|
+
|
43
|
+
# Execute a block after the given seconds, you have to call [#start] on it
|
44
|
+
# yourself.
|
45
|
+
#
|
46
|
+
# @param time [Float] the seconds after it gets called
|
47
|
+
#
|
48
|
+
# @return [Delay] the object representing the timeout
|
49
|
+
def after!(time, &block)
|
50
|
+
Delay.new(@native, time, &block)
|
51
|
+
end
|
52
|
+
|
53
|
+
# Returns a promise that will resolve after the given seconds.
|
54
|
+
#
|
55
|
+
# @param time [Float] the seconds after it gets called
|
56
|
+
#
|
57
|
+
# @return [Promise] the promise that will resolve after timeout happens
|
58
|
+
def resolve_after(time)
|
59
|
+
promise = Promise.new
|
60
|
+
Delay.new(@native, time) { promise.resolve }.start
|
61
|
+
promise
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
end
|
66
|
+
|
67
|
+
module Kernel
|
68
|
+
# (see Browser::Window#after)
|
69
|
+
def after(time, &block)
|
70
|
+
$window.after(time, &block)
|
71
|
+
end
|
72
|
+
|
73
|
+
# (see Browser::Window#after!)
|
74
|
+
def after!(time, &block)
|
75
|
+
$window.after!(time, &block)
|
76
|
+
end
|
77
|
+
|
78
|
+
# (see Browser::Window#resolve_after)
|
79
|
+
def resolve_after(time)
|
80
|
+
$window.resolve_after(time)
|
81
|
+
end
|
82
|
+
end
|
83
|
+
|
84
|
+
class Proc
|
85
|
+
# (see Browser::Window#after)
|
86
|
+
def after(time)
|
87
|
+
$window.after(time, &self)
|
88
|
+
end
|
89
|
+
|
90
|
+
# (see Browser::Window#after!)
|
91
|
+
def after!(time)
|
92
|
+
$window.after!(time, &self)
|
93
|
+
end
|
94
|
+
end
|
@@ -1,18 +1,25 @@
|
|
1
1
|
module Browser; module DOM
|
2
2
|
|
3
|
+
# Encapsulates an {Element} attribute.
|
3
4
|
class Attribute
|
4
|
-
include
|
5
|
+
include Browser::NativeCachedWrapper
|
5
6
|
|
6
|
-
|
7
|
-
|
8
|
-
|
7
|
+
# @!attribute [r] name
|
8
|
+
# @return [String] the name of the attribute
|
9
|
+
alias_native :name
|
9
10
|
|
10
|
-
|
11
|
-
|
12
|
-
|
11
|
+
# @!attribute value
|
12
|
+
# @return [String] the value of the attribute
|
13
|
+
alias_native :value
|
14
|
+
alias_native :value=
|
13
15
|
|
14
|
-
|
15
|
-
|
16
|
+
# Returns true if the attribute is an id.
|
17
|
+
if Browser.supports? 'Attr.isId'
|
18
|
+
alias_native :id?, :isId
|
19
|
+
else
|
20
|
+
def id?
|
21
|
+
name == :id
|
22
|
+
end
|
16
23
|
end
|
17
24
|
end
|
18
25
|
|
data/opal/browser/dom/builder.rb
CHANGED
@@ -9,10 +9,6 @@ class Paggio::HTML::Element < BasicObject
|
|
9
9
|
def on(*args, &block)
|
10
10
|
(@on ||= []) << [args, block]
|
11
11
|
end
|
12
|
-
|
13
|
-
def style(*args, &block)
|
14
|
-
@style = [args, block]
|
15
|
-
end
|
16
12
|
end
|
17
13
|
|
18
14
|
module Browser; module DOM
|
@@ -33,23 +29,40 @@ class Builder
|
|
33
29
|
def self.build(builder, item)
|
34
30
|
to_h.each {|klass, block|
|
35
31
|
if klass === item
|
36
|
-
|
32
|
+
return block.call(builder, item)
|
37
33
|
end
|
38
34
|
}
|
35
|
+
|
36
|
+
raise ArgumentError, "cannot build unknown item #{item}"
|
39
37
|
end
|
40
38
|
|
41
39
|
attr_reader :document, :element
|
42
40
|
|
43
|
-
|
41
|
+
NEW_PAGGIO = (Paggio::HTML.instance_method(:build!) rescue false)
|
42
|
+
|
43
|
+
def initialize(document, builder=nil, &block)
|
44
44
|
@document = document
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
45
|
+
|
46
|
+
# Compatibility issue due to an unreleased Paggio gem.
|
47
|
+
# Let's try to support both versions. When Paggio is released,
|
48
|
+
# we may remove it.
|
49
|
+
|
50
|
+
if NEW_PAGGIO
|
51
|
+
@builder = Paggio::HTML.new(defer: true, &block)
|
52
|
+
|
53
|
+
build = proc do
|
54
|
+
@builder.build!(force_call: !!builder)
|
55
|
+
@roots = @builder.each.map { |e| Builder.build(self, e) }
|
56
|
+
end
|
57
|
+
|
58
|
+
if builder
|
59
|
+
builder.extend!(@builder, &build)
|
60
|
+
else
|
61
|
+
build.()
|
62
|
+
end
|
63
|
+
else
|
64
|
+
@builder = Paggio::HTML.new(&block)
|
65
|
+
@roots = @builder.each.map { |e| Builder.build(self, e) }
|
53
66
|
end
|
54
67
|
end
|
55
68
|
|
@@ -63,15 +76,12 @@ Builder.for String do |b, item|
|
|
63
76
|
end
|
64
77
|
|
65
78
|
Builder.for Paggio::HTML::Element do |b, item|
|
66
|
-
|
79
|
+
options = {}
|
67
80
|
|
68
|
-
if Hash === `item.attributes`
|
69
|
-
|
70
|
-
end
|
81
|
+
options[:attrs] = `item.attributes` if Hash === `item.attributes`
|
82
|
+
options[:classes] = `item.class_names`
|
71
83
|
|
72
|
-
`item.
|
73
|
-
dom.add_class value
|
74
|
-
}
|
84
|
+
dom = b.document.create_element(`item.name`, **options)
|
75
85
|
|
76
86
|
if on = `item.on || nil`
|
77
87
|
on.each {|args, block|
|
@@ -79,10 +89,6 @@ Builder.for Paggio::HTML::Element do |b, item|
|
|
79
89
|
}
|
80
90
|
end
|
81
91
|
|
82
|
-
if style = `item.style || nil`
|
83
|
-
dom.style(*style[0], &style[1])
|
84
|
-
end
|
85
|
-
|
86
92
|
if inner = `item.inner_html || nil`
|
87
93
|
dom.inner_html = inner
|
88
94
|
else
|
@@ -94,4 +100,8 @@ Builder.for Paggio::HTML::Element do |b, item|
|
|
94
100
|
dom
|
95
101
|
end
|
96
102
|
|
103
|
+
Builder.for DOM::Node do |b, item|
|
104
|
+
item
|
105
|
+
end
|
106
|
+
|
97
107
|
end; end
|
@@ -1,34 +1,70 @@
|
|
1
1
|
module Browser; module DOM
|
2
2
|
|
3
3
|
class CharacterData < Node
|
4
|
+
# Append data to the node.
|
5
|
+
#
|
6
|
+
# @param string [String] the data to add
|
7
|
+
#
|
8
|
+
# @return [self]
|
9
|
+
def append(string)
|
10
|
+
`#@native.appendData(string)`
|
11
|
+
|
12
|
+
self
|
13
|
+
end
|
14
|
+
|
15
|
+
# @!attribute [r] data
|
16
|
+
# @return [String] the data of the node
|
4
17
|
def data
|
5
18
|
`#@native.data`
|
6
19
|
end
|
7
20
|
|
8
|
-
|
9
|
-
|
21
|
+
# Delete data from the node.
|
22
|
+
#
|
23
|
+
# @param count [Integer] how much data to delete
|
24
|
+
# @param offset [Integer] the offset to start at
|
25
|
+
#
|
26
|
+
# @return [self]
|
27
|
+
def delete(count, offset = 0)
|
28
|
+
`#@native.deleteData(offset, count)`
|
10
29
|
|
11
30
|
self
|
12
31
|
end
|
13
32
|
|
33
|
+
# Insert data in the node.
|
34
|
+
#
|
35
|
+
# @param string [String] the data to insert
|
36
|
+
# @param offset [Integer] the offset to start at
|
37
|
+
#
|
38
|
+
# @return [self]
|
14
39
|
def insert(string, offset = 0)
|
15
40
|
`#@native.insertData(offset, string)`
|
16
41
|
|
17
42
|
self
|
18
43
|
end
|
19
44
|
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
self
|
24
|
-
end
|
45
|
+
# @!attribute [r] length
|
46
|
+
# @return [Integer] the length of the node
|
47
|
+
alias_native :length
|
25
48
|
|
49
|
+
# Replace data in the node.
|
50
|
+
#
|
51
|
+
# @param string [String] the data to replace with
|
52
|
+
# @param offset [Integer] the offset to start at
|
53
|
+
# @param count [Integer] how much data to replace
|
54
|
+
#
|
55
|
+
# @return [self]
|
26
56
|
def replace(string, offset = 0, count = `#@native.length`)
|
27
57
|
`#@native.replaceData(offset, count, string)`
|
28
58
|
|
29
59
|
self
|
30
60
|
end
|
31
61
|
|
62
|
+
# Get a substring of the data.
|
63
|
+
#
|
64
|
+
# @param count [Integer] how much data to lice
|
65
|
+
# @param offset [Integer] the offset to start at
|
66
|
+
#
|
67
|
+
# @return [String] the substring
|
32
68
|
def substring(count, offset = 0)
|
33
69
|
`#@native.substringData(offset, count)`
|
34
70
|
end
|