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
@@ -1,23 +1,47 @@
|
|
1
1
|
module Browser; module DOM
|
2
2
|
|
3
|
+
# A {MutationObserver} is a performant way to observe changes in the DOM,
|
4
|
+
# either on the tree, the attributes or data.
|
5
|
+
#
|
6
|
+
# @see https://developer.mozilla.org/en/docs/Web/API/MutationObserver
|
3
7
|
class MutationObserver
|
4
|
-
|
8
|
+
def self.supported?
|
9
|
+
Browser.supports? :MutationObserver
|
10
|
+
end
|
11
|
+
|
12
|
+
include Native::Wrapper
|
5
13
|
|
14
|
+
# Encapsulates a recorded change.
|
6
15
|
class Record
|
7
|
-
include
|
16
|
+
include Browser::NativeCachedWrapper
|
8
17
|
|
18
|
+
# @!attribute [r] type
|
19
|
+
# @return [:attributes, :tree, :cdata] the type of the recorded change
|
9
20
|
def type
|
10
21
|
case `#@native.type`
|
11
|
-
when :attributes then :
|
22
|
+
when :attributes then :attribute
|
12
23
|
when :childList then :tree
|
13
24
|
when :characterData then :cdata
|
14
25
|
end
|
15
26
|
end
|
16
27
|
|
17
|
-
|
18
|
-
def
|
19
|
-
|
28
|
+
# Returns true if the change happened on attributes.
|
29
|
+
def attribute?
|
30
|
+
type == :attribute
|
31
|
+
end
|
32
|
+
|
33
|
+
# Returns true if the change happened on the tree.
|
34
|
+
def tree?
|
35
|
+
type == :tree
|
36
|
+
end
|
20
37
|
|
38
|
+
# Returns true if the change happened in a CDATA section.
|
39
|
+
def cdata?
|
40
|
+
type == :cdata
|
41
|
+
end
|
42
|
+
|
43
|
+
# @!attribute [r] added
|
44
|
+
# @return [NodeSet] the added nodes
|
21
45
|
def added
|
22
46
|
array = if `#@native.addedNodes != null`
|
23
47
|
Native::Array.new(`#@native.addedNodes`)
|
@@ -25,9 +49,11 @@ class MutationObserver
|
|
25
49
|
[]
|
26
50
|
end
|
27
51
|
|
28
|
-
NodeSet
|
52
|
+
NodeSet[array]
|
29
53
|
end
|
30
54
|
|
55
|
+
# @!attribute [r] removed
|
56
|
+
# @return [NodeSet] the removed nodes
|
31
57
|
def removed
|
32
58
|
array = if `#@native.removedNodes != null`
|
33
59
|
Native::Array.new(`#@native.removedNodes`)
|
@@ -35,17 +61,31 @@ class MutationObserver
|
|
35
61
|
[]
|
36
62
|
end
|
37
63
|
|
38
|
-
NodeSet
|
64
|
+
NodeSet[array]
|
39
65
|
end
|
40
66
|
|
67
|
+
# @!attribute [r] target
|
68
|
+
# @return [Node] the node the mutation affected
|
41
69
|
def target
|
42
70
|
DOM(`#@native.target`)
|
43
71
|
end
|
44
72
|
|
73
|
+
# @!attribute [r] old
|
74
|
+
# @return [String] the old value
|
45
75
|
alias_native :old, :oldValue
|
46
|
-
|
76
|
+
|
77
|
+
# @!attribute [r] name
|
78
|
+
# @return [String] the name of the attribute
|
79
|
+
alias_native :name, :attributeName
|
80
|
+
|
81
|
+
# @!attribute [r] namespace
|
82
|
+
# @return [String] the namespace of the attribute
|
83
|
+
alias_native :namespace, :attributeNamespace
|
47
84
|
end
|
48
85
|
|
86
|
+
# Create a new MutationObserver with the given block.
|
87
|
+
#
|
88
|
+
# @yieldparam records [Array<Record>] the recorded changes
|
49
89
|
def initialize(&block)
|
50
90
|
%x{
|
51
91
|
var func = function(records) {
|
@@ -56,6 +96,22 @@ class MutationObserver
|
|
56
96
|
super(`new window.MutationObserver(func)`)
|
57
97
|
end
|
58
98
|
|
99
|
+
# Observe the given target with the given options.
|
100
|
+
#
|
101
|
+
# The supported options are:
|
102
|
+
#
|
103
|
+
# + **children** - whether to observe changes on the children
|
104
|
+
# of the target or not
|
105
|
+
# + **tree** - whether to observe changes on the whole subtree
|
106
|
+
# or not
|
107
|
+
# + **attributes** - whether to observe changes to attributes or not,
|
108
|
+
# if the value is `:old` the old value will be saved
|
109
|
+
# + **cdata** - whether to observe changes to CDATA sections or not,
|
110
|
+
# if the value is `:old` the old value will be saved
|
111
|
+
# + **filter** - array of attribute names to observe
|
112
|
+
#
|
113
|
+
# @param target [DOM::Node, native] the node to observe
|
114
|
+
# @param options [Hash?] the options
|
59
115
|
def observe(target, options = nil)
|
60
116
|
unless options
|
61
117
|
options = {
|
@@ -71,10 +127,14 @@ class MutationObserver
|
|
71
127
|
self
|
72
128
|
end
|
73
129
|
|
130
|
+
# Empty the observer queue and return its contents.
|
131
|
+
#
|
132
|
+
# @return [Array<Record>]
|
74
133
|
def take
|
75
134
|
`#@native.takeRecords()`.map { |r| Record.new(r) }
|
76
135
|
end
|
77
136
|
|
137
|
+
# Disconnect the observer, thus stopping observing any changes.
|
78
138
|
def disconnect
|
79
139
|
`#@native.disconnect()`
|
80
140
|
end
|
data/opal/browser/dom/node.rb
CHANGED
@@ -1,7 +1,10 @@
|
|
1
1
|
module Browser; module DOM
|
2
2
|
|
3
|
+
# Abstract class for all DOM node types.
|
4
|
+
#
|
5
|
+
# @see https://developer.mozilla.org/en-US/docs/Web/API/Node
|
3
6
|
class Node
|
4
|
-
include
|
7
|
+
include Browser::NativeCachedWrapper
|
5
8
|
|
6
9
|
ELEMENT_NODE = 1
|
7
10
|
ATTRIBUTE_NODE = 2
|
@@ -16,6 +19,11 @@ class Node
|
|
16
19
|
DOCUMENT_FRAGMENT_NODE = 11
|
17
20
|
NOTATION_NODE = 12
|
18
21
|
|
22
|
+
# Wrap a native DOM node.
|
23
|
+
#
|
24
|
+
# @param value [native] the native DOM node
|
25
|
+
#
|
26
|
+
# @return [Node]
|
19
27
|
def self.new(value)
|
20
28
|
if self == Node
|
21
29
|
@classes ||= [nil, Element, Attribute, Text, CDATA, nil, nil, nil, Comment, Document, nil, DocumentFragment]
|
@@ -30,58 +38,156 @@ class Node
|
|
30
38
|
end
|
31
39
|
end
|
32
40
|
|
33
|
-
def
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
def =~(name)
|
38
|
-
self.name.downcase == name.downcase
|
41
|
+
def initialize(node)
|
42
|
+
raise ArgumentError, "Please ensure that #initialize of #{self.class} accepts one argument" unless node
|
43
|
+
super
|
39
44
|
end
|
40
45
|
|
46
|
+
# Return true of the other element is the same underlying DOM node.
|
47
|
+
#
|
48
|
+
# @return [Boolean]
|
49
|
+
def ==(other)
|
50
|
+
`#@native === #{Native.convert(other)}`
|
51
|
+
end
|
52
|
+
|
53
|
+
# Initialize a new node after `#dup` or `#clone`.
|
54
|
+
#
|
55
|
+
# This method is not to be called directly. Use `Node#dup` or
|
56
|
+
# `Node#clone`.
|
57
|
+
#
|
58
|
+
# This method creates a deep detached clone of a DOM subtree to be used
|
59
|
+
# in the same document. The new node will have all events detached.
|
60
|
+
def initialize_copy(old)
|
61
|
+
set_native_reference `#{old.to_n}.cloneNode(true)`
|
62
|
+
end
|
63
|
+
|
64
|
+
# Append a child to the node.
|
65
|
+
#
|
66
|
+
# When passing a {String} a text node will be created.
|
67
|
+
#
|
68
|
+
# When passing an Object that responds to #each, every yielded element
|
69
|
+
# will be added following the same logic.
|
70
|
+
#
|
71
|
+
# @param node [String, Node, #each, #to_n] the node to append
|
72
|
+
#
|
73
|
+
# @return [self]
|
41
74
|
def <<(node)
|
42
|
-
|
43
|
-
|
75
|
+
if Opal.respond_to? node, :each
|
76
|
+
node.each { |n| self << n }
|
77
|
+
return self
|
78
|
+
elsif Opal.respond_to? node, :to_dom
|
79
|
+
node = node.to_dom(document)
|
80
|
+
end
|
44
81
|
|
45
|
-
|
46
|
-
|
82
|
+
unless native?(node)
|
83
|
+
if String === node
|
84
|
+
node = `#@native.ownerDocument.createTextNode(node)`
|
85
|
+
else
|
86
|
+
node = Native.convert(node)
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
`#@native.appendChild(node)`
|
91
|
+
|
92
|
+
self
|
47
93
|
end
|
48
94
|
|
49
|
-
def
|
50
|
-
if
|
95
|
+
def >>(node)
|
96
|
+
if Opal.respond_to? node, :each
|
97
|
+
node.each { |n| self >> n }
|
98
|
+
return self
|
99
|
+
elsif Opal.respond_to? node, :to_dom
|
100
|
+
node = node.to_dom(document)
|
101
|
+
end
|
102
|
+
|
103
|
+
unless native?(node)
|
104
|
+
if String === node
|
105
|
+
node = `#@native.ownerDocument.createTextNode(node)`
|
106
|
+
else
|
107
|
+
node = Native.convert(node)
|
108
|
+
end
|
109
|
+
end
|
110
|
+
|
111
|
+
if `#@native.firstChild == null`
|
51
112
|
`#@native.appendChild(node)`
|
52
|
-
elsif node.respond_to? :each
|
53
|
-
node.each { |n| add_child(n) }
|
54
|
-
elsif String === node
|
55
|
-
`#@native.appendChild(#@native.ownerDocument.createTextNode(node))`
|
56
113
|
else
|
57
|
-
`#@native.
|
114
|
+
`#@native.insertBefore(node, #@native.firstChild)`
|
58
115
|
end
|
59
116
|
|
60
117
|
self
|
61
118
|
end
|
62
119
|
|
63
|
-
def
|
64
|
-
|
120
|
+
def add_child(node = nil, &block)
|
121
|
+
unless node
|
122
|
+
node = DOM(&block)
|
123
|
+
end
|
65
124
|
|
66
|
-
self
|
125
|
+
self << node
|
67
126
|
end
|
68
127
|
|
69
|
-
|
70
|
-
|
128
|
+
# Add the passed node after this one.
|
129
|
+
#
|
130
|
+
# When passing a {String} a text node will be created.
|
131
|
+
#
|
132
|
+
# @param node [String, Node, #to_n] the node to add
|
133
|
+
def add_next_sibling(node = nil, &block)
|
134
|
+
unless node
|
135
|
+
node = DOM(&block)
|
136
|
+
end
|
137
|
+
node = node.to_dom(document) if Opal.respond_to? node, :to_dom
|
71
138
|
|
72
|
-
|
139
|
+
unless native?(node)
|
140
|
+
if String === node
|
141
|
+
node = `#@native.ownerDocument.createTextNode(node)`
|
142
|
+
else
|
143
|
+
node = Native.convert(node)
|
144
|
+
end
|
145
|
+
end
|
146
|
+
|
147
|
+
`#@native.parentNode.insertBefore(node, #@native.nextSibling)`
|
73
148
|
end
|
74
149
|
|
75
|
-
|
150
|
+
# Add the passed node before this one.
|
151
|
+
#
|
152
|
+
# When passing a {String} a text node will be created.
|
153
|
+
#
|
154
|
+
# @param node [String, Node, #to_n] the node to add
|
155
|
+
def add_previous_sibling(node = nil, &block)
|
156
|
+
unless node
|
157
|
+
node = DOM(&block)
|
158
|
+
end
|
159
|
+
node = node.to_dom(document) if Opal.respond_to? node, :to_dom
|
76
160
|
|
77
|
-
|
78
|
-
|
161
|
+
unless native?(node)
|
162
|
+
if String === node
|
163
|
+
node = `#@native.ownerDocument.createTextNode(node)`
|
164
|
+
else
|
165
|
+
node = Native.convert(node)
|
166
|
+
end
|
167
|
+
end
|
79
168
|
|
169
|
+
`#@native.parentNode.insertBefore(node, #@native)`
|
170
|
+
end
|
171
|
+
|
172
|
+
alias after add_next_sibling
|
173
|
+
|
174
|
+
# Append the node to the passed one.
|
175
|
+
#
|
176
|
+
# @param node [Node] the node to append to
|
177
|
+
def append_to(node)
|
178
|
+
node << self
|
80
179
|
self
|
81
180
|
end
|
82
181
|
|
182
|
+
# Get an array of ancestors.
|
183
|
+
#
|
184
|
+
# Passing a selector will select the ancestors matching it.
|
185
|
+
#
|
186
|
+
# @param expression [String] the selector to use as filter
|
187
|
+
#
|
188
|
+
# @return [NodeSet]
|
83
189
|
def ancestors(expression = nil)
|
84
|
-
return NodeSet
|
190
|
+
return NodeSet[] unless parent
|
85
191
|
|
86
192
|
parents = [parent]
|
87
193
|
|
@@ -93,124 +199,146 @@ class Node
|
|
93
199
|
parents.pop
|
94
200
|
end
|
95
201
|
|
96
|
-
|
202
|
+
if expression
|
203
|
+
parents.select! { |p| p =~ expression }
|
204
|
+
end
|
97
205
|
|
98
|
-
NodeSet.new
|
99
|
-
|
100
|
-
|
206
|
+
NodeSet.new(parents)
|
207
|
+
end
|
208
|
+
|
209
|
+
def attached?
|
210
|
+
`#@native.isConnected`
|
101
211
|
end
|
102
212
|
|
103
213
|
alias before add_previous_sibling
|
104
214
|
|
215
|
+
# Remove the node from its parent.
|
105
216
|
def remove
|
106
|
-
|
107
|
-
clear
|
108
|
-
|
217
|
+
parent.remove_child(self) if parent
|
109
218
|
self
|
110
219
|
end
|
111
220
|
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
self
|
221
|
+
# Remove all the children of the node.
|
222
|
+
def clear
|
223
|
+
children.remove
|
116
224
|
end
|
117
225
|
|
118
|
-
|
226
|
+
# @!attribute content
|
227
|
+
# @return [String] the inner text content of the node
|
228
|
+
if Browser.supports? 'Element.textContent'
|
229
|
+
def content
|
230
|
+
`#@native.textContent`
|
231
|
+
end
|
119
232
|
|
120
|
-
|
121
|
-
|
233
|
+
def content=(value)
|
234
|
+
`#@native.textContent = #{value}`
|
235
|
+
end
|
236
|
+
elsif Browser.supports? 'Element.innerText'
|
237
|
+
def content
|
238
|
+
`#@native.innerText`
|
239
|
+
end
|
122
240
|
|
123
|
-
|
124
|
-
|
241
|
+
def content=(value)
|
242
|
+
`#@native.innerText = #{value}`
|
243
|
+
end
|
244
|
+
else
|
245
|
+
def content
|
246
|
+
raise NotImplementedError, 'node text content unsupported'
|
247
|
+
end
|
125
248
|
|
126
|
-
|
127
|
-
|
249
|
+
def content=(value)
|
250
|
+
raise NotImplementedError, 'node text content unsupported'
|
251
|
+
end
|
128
252
|
end
|
129
253
|
|
130
254
|
def blank?
|
131
255
|
raise NotImplementedError
|
132
256
|
end
|
133
257
|
|
258
|
+
# Return true if the node is a CDATA section.
|
134
259
|
def cdata?
|
135
260
|
node_type == CDATA_SECTION_NODE
|
136
261
|
end
|
137
262
|
|
263
|
+
# @!attribute [r] child
|
264
|
+
# @return [Node?] the first child of the node
|
138
265
|
def child
|
139
266
|
children.first
|
140
267
|
end
|
141
268
|
|
269
|
+
# @!attribute children
|
270
|
+
# @return [NodeSet] the children of the node
|
142
271
|
def children
|
143
|
-
NodeSet
|
272
|
+
NodeSet[Native::Array.new(`#@native.childNodes`)]
|
144
273
|
end
|
145
274
|
|
146
275
|
def children=(node)
|
147
276
|
raise NotImplementedError
|
148
277
|
end
|
149
278
|
|
279
|
+
# Return true if the node is a comment.
|
150
280
|
def comment?
|
151
281
|
node_type == COMMENT_NODE
|
152
282
|
end
|
153
283
|
|
284
|
+
# Return true if the node is a custom element.
|
285
|
+
def custom?
|
286
|
+
false
|
287
|
+
end
|
288
|
+
|
289
|
+
# @!attribute [rw] document
|
290
|
+
# @return [Document?] the document the node is attached to
|
154
291
|
def document
|
155
|
-
DOM(`#@native.ownerDocument`)
|
292
|
+
DOM(`#@native.ownerDocument`) if defined?(`#@native.ownerDocument`)
|
156
293
|
end
|
157
294
|
|
295
|
+
# Detach a node and transfer it to another document.
|
296
|
+
def document=(new_document)
|
297
|
+
`#{Native.try_convert(new_document, new_document)}.adoptNode(#@native)`
|
298
|
+
end
|
299
|
+
|
300
|
+
# Return true if the node is a document.
|
158
301
|
def document?
|
159
302
|
node_type == DOCUMENT_NODE
|
160
303
|
end
|
161
304
|
|
305
|
+
# Return true if the node is an element.
|
162
306
|
def elem?
|
163
307
|
node_type == ELEMENT_NODE
|
164
308
|
end
|
165
309
|
|
166
310
|
alias element? elem?
|
167
311
|
|
312
|
+
# @!attribute [r] element_children
|
313
|
+
# @return [NodeSet] all the children which are elements
|
168
314
|
def element_children
|
169
315
|
children.select(&:element?)
|
170
316
|
end
|
171
317
|
|
172
318
|
alias elements element_children
|
173
319
|
|
320
|
+
# @!attribute [r] first_element_child
|
321
|
+
# @return [Element?] the first element child
|
174
322
|
def first_element_child
|
175
323
|
element_children.first
|
176
324
|
end
|
177
325
|
|
326
|
+
# Return true if the node is a document fragment.
|
178
327
|
def fragment?
|
179
328
|
node_type == DOCUMENT_FRAGMENT_NODE
|
180
329
|
end
|
181
330
|
|
182
|
-
|
183
|
-
|
184
|
-
end
|
185
|
-
|
186
|
-
def inner_html(*)
|
187
|
-
`#@native.innerHTML`
|
188
|
-
end
|
189
|
-
|
190
|
-
def inner_html=(value)
|
191
|
-
`#@native.innerHTML = #{value}`
|
192
|
-
end
|
193
|
-
|
194
|
-
def inner_text(*)
|
195
|
-
`#@native.textContent`
|
196
|
-
end
|
197
|
-
|
198
|
-
alias content inner_text
|
199
|
-
|
200
|
-
def inner_text=(value)
|
201
|
-
`#@native.textContent = #{value}`
|
202
|
-
end
|
203
|
-
|
204
|
-
alias content= inner_text=
|
331
|
+
alias inner_text content
|
332
|
+
alias inner_text= content=
|
205
333
|
|
334
|
+
# @!attribute [r] last_element_child
|
335
|
+
# @return [Element?] the last element child
|
206
336
|
def last_element_child
|
207
337
|
element_children.last
|
208
338
|
end
|
209
339
|
|
210
|
-
|
211
|
-
|
212
|
-
end
|
213
|
-
|
340
|
+
# @!attribute name
|
341
|
+
# @return [String] the name of the node
|
214
342
|
def name
|
215
343
|
`#@native.nodeName || nil`
|
216
344
|
end
|
@@ -219,14 +347,22 @@ class Node
|
|
219
347
|
`#@native.nodeName = #{value.to_s}`
|
220
348
|
end
|
221
349
|
|
350
|
+
# @!attribute [r] namespace
|
351
|
+
# @return [String] the namespace of the node
|
222
352
|
def namespace
|
223
353
|
`#@native.namespaceURI || nil`
|
224
354
|
end
|
225
355
|
|
356
|
+
# @!attribute next
|
357
|
+
# @return [Node?] the next sibling of the node
|
226
358
|
def next
|
227
359
|
DOM(`#@native.nextSibling`) if `#@native.nextSibling != null`
|
228
360
|
end
|
229
361
|
|
362
|
+
alias next= add_next_sibling
|
363
|
+
|
364
|
+
# @!attribute [r] next_element
|
365
|
+
# @return [Element?] the next element sibling of the node
|
230
366
|
def next_element
|
231
367
|
current = self.next
|
232
368
|
|
@@ -243,19 +379,31 @@ class Node
|
|
243
379
|
|
244
380
|
alias node_name= name=
|
245
381
|
|
382
|
+
# @!attribute [r] node_type
|
383
|
+
# @return [Symbol] the type of the node
|
246
384
|
def node_type
|
247
385
|
`#@native.nodeType`
|
248
386
|
end
|
249
387
|
|
388
|
+
# @!attribute outer_html
|
389
|
+
# @return [String] the simulated outer html of the node
|
390
|
+
def outer_html
|
391
|
+
div = $document.create_element("DIV")
|
392
|
+
div << self.dup
|
393
|
+
div.inner_html
|
394
|
+
end
|
395
|
+
|
396
|
+
# @!attribute parent
|
397
|
+
# @return [Element?] the parent of the node
|
250
398
|
def parent
|
251
399
|
DOM(`#@native.parentNode`) if `#@native.parentNode != null`
|
252
400
|
end
|
253
401
|
|
254
|
-
def parent=
|
255
|
-
`#@native.parentNode = #{Native.
|
402
|
+
def parent=(node)
|
403
|
+
`#@native.parentNode = #{Native.convert(node)}`
|
256
404
|
end
|
257
405
|
|
258
|
-
def parse
|
406
|
+
def parse(text, options = {})
|
259
407
|
raise NotImplementedError
|
260
408
|
end
|
261
409
|
|
@@ -263,12 +411,24 @@ class Node
|
|
263
411
|
raise NotImplementedError
|
264
412
|
end
|
265
413
|
|
414
|
+
# Prepend the node to the passed one.
|
415
|
+
#
|
416
|
+
# @param node [Node] the node to prepend to
|
417
|
+
def prepend_to(node)
|
418
|
+
node >> self
|
419
|
+
self
|
420
|
+
end
|
421
|
+
|
422
|
+
# @!attribute previous
|
423
|
+
# @return [Node?] the previous sibling of the node
|
266
424
|
def previous
|
267
425
|
DOM(`#@native.previousSibling`) if `#@native.previousSibling != null`
|
268
426
|
end
|
269
427
|
|
270
428
|
alias previous= add_previous_sibling
|
271
429
|
|
430
|
+
# @!attribute [r] previous_element
|
431
|
+
# @return [Element?] the previous element sibling of the node
|
272
432
|
def previous_element
|
273
433
|
current = self.previous
|
274
434
|
|
@@ -281,16 +441,40 @@ class Node
|
|
281
441
|
|
282
442
|
alias previous_sibling previous
|
283
443
|
|
284
|
-
#
|
444
|
+
# Remove the given node from the children of this node.
|
445
|
+
def remove_child(node)
|
446
|
+
`#@native.removeChild(#{Native.try_convert(node)})`
|
447
|
+
self
|
448
|
+
end
|
449
|
+
|
450
|
+
# Replace the node with the given one.
|
451
|
+
#
|
452
|
+
# @todo implement for NodeSet
|
453
|
+
#
|
454
|
+
# @param node [Node] the node to replace with
|
455
|
+
# @return [Node] the passed node
|
285
456
|
def replace(node)
|
286
|
-
|
457
|
+
node = node.to_dom(document) if Opal.respond_to? node, :to_dom
|
287
458
|
|
288
|
-
node
|
459
|
+
unless native?(node)
|
460
|
+
if String === node
|
461
|
+
node = `#@native.ownerDocument.createTextNode(node)`
|
462
|
+
else
|
463
|
+
node = Native.convert(node)
|
464
|
+
end
|
465
|
+
end
|
466
|
+
|
467
|
+
`#@native.parentNode.replaceChild(node, #@native)`
|
468
|
+
|
469
|
+
DOM(node)
|
289
470
|
end
|
290
471
|
|
291
|
-
alias
|
292
|
-
|
472
|
+
alias replace_with replace
|
473
|
+
|
474
|
+
alias text content
|
475
|
+
alias text= content=
|
293
476
|
|
477
|
+
# Return true if the node is a text node.
|
294
478
|
def text?
|
295
479
|
node_type == TEXT_NODE
|
296
480
|
end
|
@@ -301,6 +485,8 @@ class Node
|
|
301
485
|
|
302
486
|
alias type node_type
|
303
487
|
|
488
|
+
# @!attribute value
|
489
|
+
# @return [String] the value of the node
|
304
490
|
def value
|
305
491
|
`#@native.nodeValue || nil`
|
306
492
|
end
|
@@ -309,6 +495,7 @@ class Node
|
|
309
495
|
`#@native.nodeValue = value`
|
310
496
|
end
|
311
497
|
|
498
|
+
# @private
|
312
499
|
def inspect
|
313
500
|
"#<DOM::Node: #{name}>"
|
314
501
|
end
|