atome-opal-browser 0.3.9.5
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/CHANGELOG.md +11 -0
- data/Gemfile +33 -0
- data/Gemfile.lock +122 -0
- data/LICENSE +20 -0
- data/README.md +301 -0
- data/Rakefile +63 -0
- data/bin/rake +7 -0
- data/bin/setup +8 -0
- data/config.ru +81 -0
- data/docs/polyfills.md +24 -0
- data/examples/2048/Gemfile +13 -0
- data/examples/2048/Gemfile.lock +41 -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 +9 -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 +9 -0
- data/examples/component/Gemfile.lock +45 -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 +7 -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 +8 -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 +9 -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 +10 -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 +7 -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 +9 -0
- data/examples/integrations/static-rake-guard/Gemfile.lock +69 -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 +4 -0
- data/lib/opal-browser.rb +1 -0
- data/opal/browser/animation_frame.rb +111 -0
- 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 +63 -0
- data/opal/browser/canvas/gradient.rb +27 -0
- data/opal/browser/canvas/style.rb +115 -0
- data/opal/browser/canvas/text.rb +45 -0
- data/opal/browser/canvas.rb +335 -0
- data/opal/browser/console.rb +105 -0
- data/opal/browser/cookies.rb +171 -0
- data/opal/browser/crypto.rb +79 -0
- data/opal/browser/css/declaration.rb +83 -0
- data/opal/browser/css/rule/style.rb +16 -0
- data/opal/browser/css/rule.rb +48 -0
- data/opal/browser/css/style_sheet.rb +83 -0
- data/opal/browser/css/unit.rb +188 -0
- data/opal/browser/css.rb +40 -0
- data/opal/browser/database/sql.rb +193 -0
- data/opal/browser/delay.rb +94 -0
- data/opal/browser/dom/attribute.rb +26 -0
- data/opal/browser/dom/builder.rb +107 -0
- data/opal/browser/dom/cdata.rb +9 -0
- data/opal/browser/dom/character_data.rb +73 -0
- data/opal/browser/dom/comment.rb +9 -0
- data/opal/browser/dom/document.rb +217 -0
- data/opal/browser/dom/document_fragment.rb +25 -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 +64 -0
- data/opal/browser/dom/element/media.rb +43 -0
- data/opal/browser/dom/element/offset.rb +89 -0
- data/opal/browser/dom/element/position.rb +46 -0
- data/opal/browser/dom/element/scroll.rb +168 -0
- 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 +618 -0
- data/opal/browser/dom/mutation_observer.rb +178 -0
- data/opal/browser/dom/node.rb +504 -0
- data/opal/browser/dom/node_set.rb +121 -0
- data/opal/browser/dom/shadow_root.rb +12 -0
- data/opal/browser/dom/text.rb +36 -0
- data/opal/browser/dom.rb +124 -0
- data/opal/browser/effects.rb +216 -0
- data/opal/browser/event/all.rb +26 -0
- data/opal/browser/event/animation.rb +40 -0
- data/opal/browser/event/audio_processing.rb +35 -0
- 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/event/device_motion.rb +53 -0
- data/opal/browser/event/device_orientation.rb +50 -0
- data/opal/browser/event/device_proximity.rb +35 -0
- 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/event/hash_change.rb +30 -0
- data/opal/browser/event/keyboard.rb +128 -0
- data/opal/browser/event/message.rb +72 -0
- data/opal/browser/event/mouse.rb +258 -0
- 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/event/storage.rb +45 -0
- data/opal/browser/event/touch.rb +62 -0
- data/opal/browser/event/ui.rb +38 -0
- data/opal/browser/event/wheel.rb +51 -0
- data/opal/browser/event.rb +162 -0
- data/opal/browser/event_source.rb +70 -0
- data/opal/browser/form_data.rb +225 -0
- data/opal/browser/history.rb +86 -0
- data/opal/browser/http/binary.rb +58 -0
- data/opal/browser/http/headers.rb +109 -0
- data/opal/browser/http/request.rb +359 -0
- data/opal/browser/http/response.rb +119 -0
- data/opal/browser/http.rb +167 -0
- data/opal/browser/immediate.rb +161 -0
- data/opal/browser/interval.rb +111 -0
- data/opal/browser/location.rb +93 -0
- data/opal/browser/navigator.rb +274 -0
- data/opal/browser/polyfill/visual_viewport.rb +216 -0
- data/opal/browser/screen.rb +66 -0
- 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 +123 -0
- data/opal/browser/storage.rb +252 -0
- data/opal/browser/support.rb +299 -0
- data/opal/browser/utils.rb +154 -0
- data/opal/browser/version.rb +3 -0
- data/opal/browser/visual_viewport.rb +39 -0
- data/opal/browser/window/size.rb +73 -0
- data/opal/browser/window/view.rb +51 -0
- data/opal/browser/window.rb +133 -0
- data/opal/browser.rb +1 -0
- data/opal/opal-browser.rb +1 -0
- data/opal-browser.gemspec +27 -0
- 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 +86 -0
- data/spec/dom/document_spec.rb +62 -0
- 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 +223 -0
- data/spec/dom/mutation_observer_spec.rb +41 -0
- data/spec/dom/node_set_spec.rb +44 -0
- data/spec/dom/node_spec.rb +214 -0
- data/spec/dom_spec.rb +23 -0
- data/spec/event_source_spec.rb +45 -0
- data/spec/event_spec.rb +156 -0
- data/spec/history_spec.rb +61 -0
- data/spec/http_spec.rb +76 -0
- data/spec/immediate_spec.rb +15 -0
- data/spec/interval_spec.rb +59 -0
- data/spec/json2.js +486 -0
- data/spec/native_cached_wrapper_spec.rb +46 -0
- data/spec/runner.rb +107 -0
- data/spec/sizzle.js +5 -0
- data/spec/socket_spec.rb +47 -0
- data/spec/spec_helper.rb +35 -0
- data/spec/spec_helper_promise.rb.erb +25 -0
- data/spec/storage_spec.rb +26 -0
- data/spec/wgxpath.install.js +49 -0
- data/spec/window_spec.rb +10 -0
- metadata +500 -0
@@ -0,0 +1,31 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Button < Element
|
4
|
+
def_selector "button"
|
5
|
+
|
6
|
+
def disabled?
|
7
|
+
`#@native.disabled`
|
8
|
+
end
|
9
|
+
|
10
|
+
def disabled=(value)
|
11
|
+
`#@native.disabled = #{value}`
|
12
|
+
end
|
13
|
+
|
14
|
+
def autofocus?
|
15
|
+
`#@native.autofocus`
|
16
|
+
end
|
17
|
+
|
18
|
+
def autofocus=(value)
|
19
|
+
`#@native.autofocus = #{value}`
|
20
|
+
end
|
21
|
+
|
22
|
+
def name_
|
23
|
+
`#@native.name`
|
24
|
+
end
|
25
|
+
|
26
|
+
def name_=(value)
|
27
|
+
`#@native.name = #{value}`
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
end; end; end
|
@@ -0,0 +1,177 @@
|
|
1
|
+
# use_strict: true
|
2
|
+
# helpers: truthy
|
3
|
+
|
4
|
+
module Browser; module DOM; class Element < Node
|
5
|
+
|
6
|
+
# CustomElements implementation for opal-browser. See examples/custom_elements/.
|
7
|
+
#
|
8
|
+
# @see https://developer.mozilla.org/en-US/docs/Web/Web_Components/Using_custom_elements
|
9
|
+
# @abstract This class should not be used directly. Please extend it and implement needed methods.
|
10
|
+
class Custom < Element
|
11
|
+
# The reason why we wrap class definition with an eval is kind of selfish. I want it to work
|
12
|
+
# with opal-optimizer which doesn't support the new class syntax. I would do it with prototypes,
|
13
|
+
# but the prototypes system is so messy I gave up.
|
14
|
+
#
|
15
|
+
# Therefore, for it to be cleaned up, one of those two must happen:
|
16
|
+
# - we raise the supported ES version in Opal and we implement those ES syntax features in
|
17
|
+
# rkelly-turbo. And then we remove the polyfill.
|
18
|
+
# - we reimplement it in terms of prototypes.
|
19
|
+
%x{
|
20
|
+
var make_custom_class = Function('self,base_class',
|
21
|
+
'"use strict"; \
|
22
|
+
var klass = class extends base_class { \
|
23
|
+
constructor() { \
|
24
|
+
super(); \
|
25
|
+
self.$_dispatch_constructor(this); \
|
26
|
+
} \
|
27
|
+
connectedCallback() { \
|
28
|
+
return this.$$opal_native_cached.$attached(); \
|
29
|
+
} \
|
30
|
+
disconnectedCallback() { \
|
31
|
+
return this.$$opal_native_cached.$detached(); \
|
32
|
+
} \
|
33
|
+
adoptedCallback() { \
|
34
|
+
return this.$$opal_native_cached.$adopted(); \
|
35
|
+
} \
|
36
|
+
attributeChangedCallback(attr, from, to) { \
|
37
|
+
if (from === null) from = Opal.nil; \
|
38
|
+
if (to === null) to = Opal.nil; \
|
39
|
+
return this.$$opal_native_cached.$attribute_changed(attr, from, to); \
|
40
|
+
} \
|
41
|
+
\
|
42
|
+
static get observedAttributes() { \
|
43
|
+
return self.$observed_attributes(); \
|
44
|
+
} \
|
45
|
+
}; \
|
46
|
+
klass.$$opal_class = self; \
|
47
|
+
return klass;'
|
48
|
+
);
|
49
|
+
} if Browser.supports? 'Custom Elements' #'
|
50
|
+
|
51
|
+
module ClassMethods
|
52
|
+
if Browser.supports? 'Custom Elements'
|
53
|
+
# Defines a new custom element. This should come as the last call
|
54
|
+
# in the class definition, because at this point the methods may
|
55
|
+
# be called!
|
56
|
+
#
|
57
|
+
# @opalopt uses:_dispatch_constructor,attached,detached,adopted,attribute_changed,observed_attributes
|
58
|
+
def def_custom(tag_name, base_class: nil, extends: nil)
|
59
|
+
if `base_class !== nil`
|
60
|
+
elsif self.superclass == Custom
|
61
|
+
base_class = `HTMLElement`
|
62
|
+
elsif self.ancestors.include? Custom
|
63
|
+
base_class = `#{self.superclass}.custom_class`
|
64
|
+
else
|
65
|
+
raise ArgumentError, "You must define base_class"
|
66
|
+
end
|
67
|
+
|
68
|
+
@custom_class = `make_custom_class(self, #{base_class})`
|
69
|
+
@observed_attributes ||= []
|
70
|
+
|
71
|
+
def_selector tag_name
|
72
|
+
|
73
|
+
%x{
|
74
|
+
if ($truthy(#{extends})) customElements.define(#{tag_name}, #{@custom_class}, {extends: #{extends}});
|
75
|
+
else customElements.define(#{tag_name}, #{@custom_class});
|
76
|
+
}
|
77
|
+
end
|
78
|
+
elsif Browser.supports? 'MutationObserver'
|
79
|
+
# Can we polyfill it?
|
80
|
+
Browser::DOM::MutationObserver.new do |obs|
|
81
|
+
obs.each do |e|
|
82
|
+
target = e.target
|
83
|
+
|
84
|
+
case e.type
|
85
|
+
when :attribute
|
86
|
+
if Custom::Mixin === target && target.class.observed_attributes.include?(e.name)
|
87
|
+
target.attribute_changed(e.name, e.old, target[e.name])
|
88
|
+
end
|
89
|
+
when :tree
|
90
|
+
e.added.each { |n| n.attached_once if Custom::Mixin === n }
|
91
|
+
e.removed.each { |n| n.detached_once if Custom::Mixin === n }
|
92
|
+
end
|
93
|
+
end
|
94
|
+
end.observe($document, tree: true, children: true, attributes: :old)
|
95
|
+
end
|
96
|
+
|
97
|
+
unless Browser.supports? 'Custom Elements'
|
98
|
+
# The polyfilled implementation. Define the selector and then
|
99
|
+
# try to upgrade the elements that are already in the document.
|
100
|
+
def def_custom(tag_name, base_class: nil, extends: nil)
|
101
|
+
def_selector tag_name
|
102
|
+
|
103
|
+
$document.body.css(tag_name).each do |elem|
|
104
|
+
_dispatch_constructor(elem.to_n)&.attached_once
|
105
|
+
end
|
106
|
+
end
|
107
|
+
end
|
108
|
+
|
109
|
+
private def _dispatch_constructor(obj)
|
110
|
+
%x{
|
111
|
+
if (typeof obj.$$opal_native_cached !== 'undefined') {
|
112
|
+
delete obj.$$opal_native_cached;
|
113
|
+
return self.$new(obj);
|
114
|
+
}
|
115
|
+
else {
|
116
|
+
self.$new(obj);
|
117
|
+
return nil;
|
118
|
+
}
|
119
|
+
}
|
120
|
+
end
|
121
|
+
|
122
|
+
# This must be defined before def_custom is called!
|
123
|
+
attr_accessor :observed_attributes
|
124
|
+
|
125
|
+
attr_reader :custom_class
|
126
|
+
end
|
127
|
+
|
128
|
+
module Mixin
|
129
|
+
def self.included(klass)
|
130
|
+
klass.extend ClassMethods
|
131
|
+
end
|
132
|
+
|
133
|
+
# @abstract
|
134
|
+
def attached
|
135
|
+
end
|
136
|
+
|
137
|
+
# @abstract
|
138
|
+
def detached
|
139
|
+
end
|
140
|
+
|
141
|
+
# @abstract
|
142
|
+
def adopted
|
143
|
+
end
|
144
|
+
|
145
|
+
# Note: for this method to fire, you will need to define
|
146
|
+
# the observed attributes.
|
147
|
+
#
|
148
|
+
# @abstract
|
149
|
+
def attribute_changed(attr, from, to)
|
150
|
+
end
|
151
|
+
|
152
|
+
# Return true if the node is a custom element.
|
153
|
+
def custom?
|
154
|
+
true
|
155
|
+
end
|
156
|
+
|
157
|
+
# Those methods keep track of the attachment status of the elements,
|
158
|
+
# so that #attached/#detached isn't called twice.
|
159
|
+
unless Browser.supports? 'Custom Elements'
|
160
|
+
# @private
|
161
|
+
def attached_once
|
162
|
+
attached unless @_polyfill_attached
|
163
|
+
@_polyfill_attached = true
|
164
|
+
end
|
165
|
+
|
166
|
+
# @private
|
167
|
+
def detached_once
|
168
|
+
detached if @_polyfill_attached
|
169
|
+
@_polyfill_attached = false
|
170
|
+
end
|
171
|
+
end
|
172
|
+
end
|
173
|
+
|
174
|
+
include Mixin
|
175
|
+
end
|
176
|
+
|
177
|
+
end; end; end
|
@@ -0,0 +1,82 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Data
|
4
|
+
attr_reader :element
|
5
|
+
|
6
|
+
def initialize(element)
|
7
|
+
@element = element
|
8
|
+
@native = element.to_n
|
9
|
+
|
10
|
+
unless defined?(`#@native.$data`)
|
11
|
+
`#@native.$data = {}`
|
12
|
+
end
|
13
|
+
end
|
14
|
+
|
15
|
+
include Enumerable
|
16
|
+
|
17
|
+
def each(&block)
|
18
|
+
return enum_for :each unless block
|
19
|
+
|
20
|
+
%x{
|
21
|
+
var data = #@native.$data;
|
22
|
+
|
23
|
+
for (var key in data) {
|
24
|
+
#{block.call `key`, `data[key]`};
|
25
|
+
}
|
26
|
+
}
|
27
|
+
|
28
|
+
@element.attributes.each {|name, value|
|
29
|
+
if name =~ /^data-(.*)$/
|
30
|
+
block.call $1, value
|
31
|
+
end
|
32
|
+
}
|
33
|
+
|
34
|
+
self
|
35
|
+
end
|
36
|
+
|
37
|
+
def assign(data)
|
38
|
+
data.each {|name, value|
|
39
|
+
self[name] = value
|
40
|
+
}
|
41
|
+
|
42
|
+
self
|
43
|
+
end
|
44
|
+
|
45
|
+
def [](name)
|
46
|
+
if data = @element["data-#{name}"]
|
47
|
+
return data
|
48
|
+
end
|
49
|
+
|
50
|
+
%x{
|
51
|
+
var value = #@native.$data[name];
|
52
|
+
|
53
|
+
if (value === undefined) {
|
54
|
+
return nil;
|
55
|
+
}
|
56
|
+
else {
|
57
|
+
return value;
|
58
|
+
}
|
59
|
+
}
|
60
|
+
end
|
61
|
+
|
62
|
+
def []=(name, value)
|
63
|
+
`delete #@native.$data[name]`
|
64
|
+
if [true, false, nil].include?(value)
|
65
|
+
@element["data-#{name}"] = value
|
66
|
+
elsif value.respond_to? :to_str
|
67
|
+
@element["data-#{name}"] = value.to_str
|
68
|
+
elsif value.respond_to? :to_int
|
69
|
+
@element["data-#{name}"] = value.to_int.to_s
|
70
|
+
else
|
71
|
+
`#@native.$data[name] = value`
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
def delete(name)
|
76
|
+
data = self[name]
|
77
|
+
self[name] = nil
|
78
|
+
data
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
end; end; end
|
@@ -0,0 +1,47 @@
|
|
1
|
+
module Browser; module DOM
|
2
|
+
|
3
|
+
class Element < Node
|
4
|
+
# @!attribute editable
|
5
|
+
# @return [Boolean?] the value of contentEditable for this element
|
6
|
+
def editable
|
7
|
+
case `#@native.contentEditable`
|
8
|
+
when "true"
|
9
|
+
true
|
10
|
+
when "false"
|
11
|
+
false
|
12
|
+
when "inherit"
|
13
|
+
nil
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
def editable=(value)
|
18
|
+
value = case value
|
19
|
+
when true
|
20
|
+
"true"
|
21
|
+
when false
|
22
|
+
"false"
|
23
|
+
when nil
|
24
|
+
"inherit"
|
25
|
+
end
|
26
|
+
`#@native.contentEditable = #{value}`
|
27
|
+
end
|
28
|
+
|
29
|
+
def editable?
|
30
|
+
`#@native.isContentEditable`
|
31
|
+
end
|
32
|
+
|
33
|
+
# Execute a contentEditable command
|
34
|
+
def edit(command, value=nil)
|
35
|
+
command = command.gsub(/_./) { |i| i[1].upcase }
|
36
|
+
|
37
|
+
focus
|
38
|
+
|
39
|
+
if value
|
40
|
+
`#{document}.native.execCommand(#{command}, false, #{value})`
|
41
|
+
else
|
42
|
+
`#{document}.native.execCommand(#{command}, false)`
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
end; end
|
@@ -0,0 +1,38 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Form < Element
|
4
|
+
def_selector "form"
|
5
|
+
|
6
|
+
# Capture the content of this form to a new {FormData} object,
|
7
|
+
#
|
8
|
+
# @return [FormData]
|
9
|
+
def form_data
|
10
|
+
FormData.create(self)
|
11
|
+
end
|
12
|
+
|
13
|
+
# Submit a form. This will fire a submit event.
|
14
|
+
def submit
|
15
|
+
`#@native.submit()`
|
16
|
+
end
|
17
|
+
|
18
|
+
# Reset a form. This will fire a reset event.
|
19
|
+
def reset
|
20
|
+
`#@native.reset()`
|
21
|
+
end
|
22
|
+
|
23
|
+
alias_native :action
|
24
|
+
alias_native :action=
|
25
|
+
alias_native :method
|
26
|
+
alias_native :method=
|
27
|
+
alias_native :target
|
28
|
+
alias_native :target=
|
29
|
+
alias_native :encoding
|
30
|
+
alias_native :encoding=
|
31
|
+
|
32
|
+
# Return a NodeSet containing all form controls belonging to this form element.
|
33
|
+
def controls
|
34
|
+
NodeSet[Native::Array.new(`#@native.elements`)]
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
end; end; end
|
@@ -0,0 +1,37 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Iframe < Element
|
4
|
+
def_selector "iframe"
|
5
|
+
|
6
|
+
# @!attribute src
|
7
|
+
# @return [String] the URL of the page to embed
|
8
|
+
alias_native :src
|
9
|
+
alias_native :src=
|
10
|
+
|
11
|
+
# @!attribute [r] content_window
|
12
|
+
# @return [Window] window of content of this iframe
|
13
|
+
def content_window
|
14
|
+
Browser::Window.new(`#@native.contentWindow`)
|
15
|
+
end
|
16
|
+
|
17
|
+
# @!attribute [r] content_document
|
18
|
+
# @return [Document] document of content of this iframe
|
19
|
+
def content_document
|
20
|
+
DOM(`#@native.contentDocument || #@native.contentWindow.document`)
|
21
|
+
end
|
22
|
+
|
23
|
+
# Send a message to the iframe content's window.
|
24
|
+
#
|
25
|
+
# @param message [String] the message
|
26
|
+
# @param options [Hash] optional `to: target`
|
27
|
+
def send(message, options={})
|
28
|
+
content_window.send(message, options)
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
# Object is not an iframe, but acts the same.
|
33
|
+
class Object < Iframe
|
34
|
+
def_selector "object"
|
35
|
+
end
|
36
|
+
|
37
|
+
end; end; end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Image < Element
|
4
|
+
def_selector "img"
|
5
|
+
|
6
|
+
def complete?
|
7
|
+
`#@native.complete`
|
8
|
+
end
|
9
|
+
|
10
|
+
def cross?
|
11
|
+
`#@native.crossOrigin`
|
12
|
+
end
|
13
|
+
|
14
|
+
def height
|
15
|
+
`#@native.naturalHeight`
|
16
|
+
end
|
17
|
+
|
18
|
+
def width
|
19
|
+
`#@native.naturalWidth`
|
20
|
+
end
|
21
|
+
end
|
22
|
+
|
23
|
+
Img = Image
|
24
|
+
|
25
|
+
end; end; end
|
@@ -0,0 +1,64 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Input < Element
|
4
|
+
def_selector "input"
|
5
|
+
|
6
|
+
def value
|
7
|
+
%x{
|
8
|
+
if (#@native.value == "") {
|
9
|
+
return nil;
|
10
|
+
}
|
11
|
+
else {
|
12
|
+
return #@native.value;
|
13
|
+
}
|
14
|
+
}
|
15
|
+
end
|
16
|
+
|
17
|
+
def value=(value)
|
18
|
+
`#@native.value = #{value}`
|
19
|
+
end
|
20
|
+
|
21
|
+
def name_
|
22
|
+
`#@native.name`
|
23
|
+
end
|
24
|
+
|
25
|
+
def type
|
26
|
+
`#@native.type`
|
27
|
+
end
|
28
|
+
|
29
|
+
def checked?
|
30
|
+
`#@native.checked`
|
31
|
+
end
|
32
|
+
|
33
|
+
def check!
|
34
|
+
`#@native.checked = 'checked'`
|
35
|
+
end
|
36
|
+
|
37
|
+
def uncheck!
|
38
|
+
`#@native.checked = ''`
|
39
|
+
end
|
40
|
+
|
41
|
+
def enabled?
|
42
|
+
`#@native.enabled`
|
43
|
+
end
|
44
|
+
|
45
|
+
def disable!
|
46
|
+
`#@native.disabled = 'disabled'`
|
47
|
+
end
|
48
|
+
|
49
|
+
def enable!
|
50
|
+
`#@native.disabled = ''`
|
51
|
+
end
|
52
|
+
|
53
|
+
def clear
|
54
|
+
`#@native.value = ''`
|
55
|
+
end
|
56
|
+
|
57
|
+
# @!attribute [r] files
|
58
|
+
# @return [Array<File>] list of files attached to this {Input}
|
59
|
+
def files
|
60
|
+
Native::Array.new(`#@native.files`).map { |f| File.new(f.to_n) }
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
end; end; end
|
@@ -0,0 +1,43 @@
|
|
1
|
+
module Browser
|
2
|
+
|
3
|
+
module DOM
|
4
|
+
|
5
|
+
class Element < Node
|
6
|
+
|
7
|
+
class Media < Element
|
8
|
+
def play
|
9
|
+
`#@native.play()`
|
10
|
+
end
|
11
|
+
|
12
|
+
def pause
|
13
|
+
`#@native.pause()`
|
14
|
+
end
|
15
|
+
|
16
|
+
def muted
|
17
|
+
`#@native.muted=true`
|
18
|
+
end
|
19
|
+
|
20
|
+
def unmuted
|
21
|
+
`#@native.muted=false`
|
22
|
+
end
|
23
|
+
|
24
|
+
def currentTime= (val)
|
25
|
+
`#@native.currentTime = #{val}`
|
26
|
+
end
|
27
|
+
|
28
|
+
def currentTime
|
29
|
+
`#@native.currentTime`
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
class Video < Media
|
34
|
+
def_selector "video"
|
35
|
+
end
|
36
|
+
|
37
|
+
class Audio < Media
|
38
|
+
def_selector "audio"
|
39
|
+
end
|
40
|
+
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
@@ -0,0 +1,89 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Offset
|
4
|
+
attr_reader :element
|
5
|
+
|
6
|
+
# @private
|
7
|
+
def initialize(element)
|
8
|
+
@element = element
|
9
|
+
@native = element.to_n
|
10
|
+
end
|
11
|
+
|
12
|
+
def parent
|
13
|
+
DOM(`#@native.offsetParent || #{@element.document.root.to_n}`)
|
14
|
+
end
|
15
|
+
|
16
|
+
# @!attribute x
|
17
|
+
# @return [Integer]
|
18
|
+
def x
|
19
|
+
get.x
|
20
|
+
end
|
21
|
+
|
22
|
+
def x=(value)
|
23
|
+
set value, nil
|
24
|
+
end
|
25
|
+
|
26
|
+
# @!attribute y
|
27
|
+
# @return [Integer]
|
28
|
+
def y
|
29
|
+
get.y
|
30
|
+
end
|
31
|
+
|
32
|
+
def y=(value)
|
33
|
+
set nil, value
|
34
|
+
end
|
35
|
+
|
36
|
+
if Browser.supports? 'Element.getBoundingClientRect'
|
37
|
+
def get
|
38
|
+
doc = @element.document
|
39
|
+
root = doc.root.to_n
|
40
|
+
win = doc.window.to_n
|
41
|
+
|
42
|
+
%x{
|
43
|
+
var box = #@native.getBoundingClientRect(),
|
44
|
+
y = box.top + (#{win}.pageYOffset || #{root}.scrollTop) - (#{root}.clientTop || 0),
|
45
|
+
x = box.left + (#{win}.pageXOffset || #{root}.scrollLeft) - (#{root}.clientLeft || 0);
|
46
|
+
}
|
47
|
+
|
48
|
+
Browser::Position.new(`x`, `y`)
|
49
|
+
end
|
50
|
+
else
|
51
|
+
def get
|
52
|
+
doc = document
|
53
|
+
root = doc.root.to_n
|
54
|
+
win = doc.window.to_n
|
55
|
+
|
56
|
+
%x{
|
57
|
+
var y = (#{win}.pageYOffset || #{root}.scrollTop) - (#{root}.clientTop || 0),
|
58
|
+
x = (#{win}.pageXOffset || #{root}.scrollLeft) - (#{root}.clientLeft || 0);
|
59
|
+
}
|
60
|
+
|
61
|
+
Browser::Position.new(`x`, `y`)
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
def set(*value)
|
66
|
+
position = @element.style![:position]
|
67
|
+
|
68
|
+
if position == :static
|
69
|
+
@element.style[:position] = :relative
|
70
|
+
end
|
71
|
+
|
72
|
+
offset = get
|
73
|
+
top = @element.style![:top].to_u
|
74
|
+
left = @element.style![:left].to_u
|
75
|
+
|
76
|
+
if Browser::Position === value.first
|
77
|
+
x, y = value.first.x, value.first.y
|
78
|
+
elsif Hash === value.first
|
79
|
+
x, y = value.first[:x], value.first[:y]
|
80
|
+
else
|
81
|
+
x, y = value
|
82
|
+
end
|
83
|
+
|
84
|
+
@element.style[:left] = (x.px - offset.x) + left if x
|
85
|
+
@element.style[:top] = (y.px - offset.y) + top if y
|
86
|
+
end
|
87
|
+
end
|
88
|
+
|
89
|
+
end; end; end
|
@@ -0,0 +1,46 @@
|
|
1
|
+
module Browser; module DOM; class Element < Node
|
2
|
+
|
3
|
+
class Position
|
4
|
+
attr_reader :element
|
5
|
+
|
6
|
+
# @private
|
7
|
+
def initialize(element)
|
8
|
+
@element = element
|
9
|
+
@native = element.to_n
|
10
|
+
end
|
11
|
+
|
12
|
+
# @private
|
13
|
+
def get
|
14
|
+
offset = @element.offset
|
15
|
+
position = offset.get
|
16
|
+
parent = offset.parent
|
17
|
+
parent_offset = Browser::Position.new(0, 0)
|
18
|
+
|
19
|
+
if @element.style[:position] == :fixed
|
20
|
+
unless parent =~ :html
|
21
|
+
parent_offset = parent.offset
|
22
|
+
end
|
23
|
+
|
24
|
+
parent_offset.x += parent.style['border-top-width'].to_i
|
25
|
+
parent_offset.y += parent.style['border-left-width'].to_i
|
26
|
+
end
|
27
|
+
|
28
|
+
Browser::Position.new(
|
29
|
+
position.x - parent_offset.x - @element.style['margin-left'].to_i,
|
30
|
+
position.y - parent_offset.y - @element.style['margin-top'].to_i)
|
31
|
+
end
|
32
|
+
|
33
|
+
# @!attribute [r] x
|
34
|
+
# @return [Integer] the position of the element on the x axis
|
35
|
+
def x
|
36
|
+
get.x
|
37
|
+
end
|
38
|
+
|
39
|
+
# @!attribute [r] y
|
40
|
+
# @return [Integer] the position of the element on the y axis
|
41
|
+
def y
|
42
|
+
get.y
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
end; end; end
|