puppeteer-ruby 0.0.10 → 0.0.15

Sign up to get free protection for your applications and to get access to all the features.
Files changed (132) hide show
  1. checksums.yaml +4 -4
  2. data/.circleci/config.yml +0 -12
  3. data/.github/workflows/docs.yml +45 -0
  4. data/.github/workflows/reviewdog.yml +15 -0
  5. data/README.md +54 -1
  6. data/lib/puppeteer.rb +37 -13
  7. data/lib/puppeteer/browser.rb +39 -26
  8. data/lib/puppeteer/cdp_session.rb +3 -19
  9. data/lib/puppeteer/concurrent_ruby_utils.rb +12 -2
  10. data/lib/puppeteer/connection.rb +16 -16
  11. data/lib/puppeteer/define_async_method.rb +23 -0
  12. data/lib/puppeteer/dom_world.rb +58 -71
  13. data/lib/puppeteer/element_handle.rb +15 -44
  14. data/lib/puppeteer/emulation_manager.rb +2 -6
  15. data/lib/puppeteer/event_callbackable.rb +11 -0
  16. data/lib/puppeteer/execution_context.rb +1 -6
  17. data/lib/puppeteer/frame.rb +38 -3
  18. data/lib/puppeteer/frame_manager.rb +7 -25
  19. data/lib/puppeteer/js_handle.rb +3 -12
  20. data/lib/puppeteer/keyboard.rb +6 -27
  21. data/lib/puppeteer/launcher.rb +6 -6
  22. data/lib/puppeteer/launcher/chrome.rb +10 -8
  23. data/lib/puppeteer/mouse.rb +8 -33
  24. data/lib/puppeteer/page.rb +132 -121
  25. data/lib/puppeteer/page/pdf_options.rb +166 -0
  26. data/lib/puppeteer/remote_object.rb +2 -5
  27. data/lib/puppeteer/target.rb +1 -1
  28. data/lib/puppeteer/touch_screen.rb +2 -7
  29. data/lib/puppeteer/version.rb +1 -1
  30. data/lib/puppeteer/wait_task.rb +2 -4
  31. data/lib/puppeteer/web_socket_transport.rb +8 -8
  32. data/puppeteer-ruby.gemspec +1 -1
  33. data/puppeteer-ruby.png +0 -0
  34. metadata +9 -102
  35. data/Dockerfile +0 -6
  36. data/docker-compose.yml +0 -15
  37. data/docs/Puppeteer.html +0 -2020
  38. data/docs/Puppeteer/AsyncAwaitBehavior.html +0 -105
  39. data/docs/Puppeteer/Browser.html +0 -2148
  40. data/docs/Puppeteer/BrowserContext.html +0 -809
  41. data/docs/Puppeteer/BrowserFetcher.html +0 -214
  42. data/docs/Puppeteer/BrowserRunner.html +0 -914
  43. data/docs/Puppeteer/BrowserRunner/BrowserProcess.html +0 -477
  44. data/docs/Puppeteer/CDPSession.html +0 -813
  45. data/docs/Puppeteer/CDPSession/Error.html +0 -124
  46. data/docs/Puppeteer/ConcurrentRubyUtils.html +0 -430
  47. data/docs/Puppeteer/Connection.html +0 -960
  48. data/docs/Puppeteer/Connection/MessageCallback.html +0 -434
  49. data/docs/Puppeteer/Connection/ProtocolError.html +0 -216
  50. data/docs/Puppeteer/Connection/RequestDebugPrinter.html +0 -217
  51. data/docs/Puppeteer/Connection/ResponseDebugPrinter.html +0 -244
  52. data/docs/Puppeteer/ConsoleMessage.html +0 -565
  53. data/docs/Puppeteer/ConsoleMessage/Location.html +0 -433
  54. data/docs/Puppeteer/DOMWorld.html +0 -2219
  55. data/docs/Puppeteer/DOMWorld/DetachedError.html +0 -124
  56. data/docs/Puppeteer/DOMWorld/DocumentEvaluationError.html +0 -124
  57. data/docs/Puppeteer/DebugPrint.html +0 -233
  58. data/docs/Puppeteer/Device.html +0 -470
  59. data/docs/Puppeteer/Devices.html +0 -139
  60. data/docs/Puppeteer/ElementHandle.html +0 -2542
  61. data/docs/Puppeteer/ElementHandle/ElementNotFoundError.html +0 -206
  62. data/docs/Puppeteer/ElementHandle/ElementNotVisibleError.html +0 -206
  63. data/docs/Puppeteer/ElementHandle/Point.html +0 -492
  64. data/docs/Puppeteer/ElementHandle/ScrollIntoViewError.html +0 -124
  65. data/docs/Puppeteer/EmulationManager.html +0 -454
  66. data/docs/Puppeteer/EventCallbackable.html +0 -433
  67. data/docs/Puppeteer/EventCallbackable/EventListeners.html +0 -435
  68. data/docs/Puppeteer/ExecutionContext.html +0 -998
  69. data/docs/Puppeteer/ExecutionContext/EvaluationError.html +0 -124
  70. data/docs/Puppeteer/ExecutionContext/JavaScriptExpression.html +0 -357
  71. data/docs/Puppeteer/ExecutionContext/JavaScriptFunction.html +0 -389
  72. data/docs/Puppeteer/FileChooser.html +0 -455
  73. data/docs/Puppeteer/Frame.html +0 -3677
  74. data/docs/Puppeteer/FrameManager.html +0 -2410
  75. data/docs/Puppeteer/FrameManager/NavigationError.html +0 -124
  76. data/docs/Puppeteer/IfPresent.html +0 -222
  77. data/docs/Puppeteer/JSHandle.html +0 -1352
  78. data/docs/Puppeteer/Keyboard.html +0 -1557
  79. data/docs/Puppeteer/Keyboard/KeyDefinition.html +0 -831
  80. data/docs/Puppeteer/Keyboard/KeyDescription.html +0 -603
  81. data/docs/Puppeteer/Launcher.html +0 -237
  82. data/docs/Puppeteer/Launcher/Base.html +0 -385
  83. data/docs/Puppeteer/Launcher/Base/ExecutablePathNotFound.html +0 -124
  84. data/docs/Puppeteer/Launcher/BrowserOptions.html +0 -441
  85. data/docs/Puppeteer/Launcher/Chrome.html +0 -669
  86. data/docs/Puppeteer/Launcher/Chrome/DefaultArgs.html +0 -382
  87. data/docs/Puppeteer/Launcher/ChromeArgOptions.html +0 -531
  88. data/docs/Puppeteer/Launcher/LaunchOptions.html +0 -893
  89. data/docs/Puppeteer/LifecycleWatcher.html +0 -834
  90. data/docs/Puppeteer/LifecycleWatcher/ExpectedLifecycle.html +0 -363
  91. data/docs/Puppeteer/LifecycleWatcher/FrameDetachedError.html +0 -206
  92. data/docs/Puppeteer/LifecycleWatcher/TerminatedError.html +0 -124
  93. data/docs/Puppeteer/Mouse.html +0 -1105
  94. data/docs/Puppeteer/Mouse/Button.html +0 -136
  95. data/docs/Puppeteer/NetworkManager.html +0 -901
  96. data/docs/Puppeteer/NetworkManager/Credentials.html +0 -385
  97. data/docs/Puppeteer/Page.html +0 -5970
  98. data/docs/Puppeteer/Page/FileChooserTimeoutError.html +0 -206
  99. data/docs/Puppeteer/Page/ScreenshotOptions.html +0 -845
  100. data/docs/Puppeteer/Page/ScriptTag.html +0 -555
  101. data/docs/Puppeteer/Page/StyleTag.html +0 -448
  102. data/docs/Puppeteer/Page/TargetCrashedError.html +0 -124
  103. data/docs/Puppeteer/RemoteObject.html +0 -1087
  104. data/docs/Puppeteer/Target.html +0 -1336
  105. data/docs/Puppeteer/Target/InitializeFailure.html +0 -124
  106. data/docs/Puppeteer/Target/TargetInfo.html +0 -729
  107. data/docs/Puppeteer/TimeoutError.html +0 -135
  108. data/docs/Puppeteer/TimeoutSettings.html +0 -496
  109. data/docs/Puppeteer/TouchScreen.html +0 -464
  110. data/docs/Puppeteer/Viewport.html +0 -837
  111. data/docs/Puppeteer/WaitTask.html +0 -637
  112. data/docs/Puppeteer/WaitTask/TerminatedError.html +0 -124
  113. data/docs/Puppeteer/WaitTask/TimeoutError.html +0 -206
  114. data/docs/Puppeteer/WebSocket.html +0 -673
  115. data/docs/Puppeteer/WebSocket/DriverImpl.html +0 -412
  116. data/docs/Puppeteer/WebSocketTransport.html +0 -600
  117. data/docs/Puppeteer/WebSocktTransportError.html +0 -124
  118. data/docs/_index.html +0 -823
  119. data/docs/class_list.html +0 -51
  120. data/docs/css/common.css +0 -1
  121. data/docs/css/full_list.css +0 -58
  122. data/docs/css/style.css +0 -496
  123. data/docs/file.README.html +0 -123
  124. data/docs/file_list.html +0 -56
  125. data/docs/frames.html +0 -17
  126. data/docs/index.html +0 -123
  127. data/docs/js/app.js +0 -314
  128. data/docs/js/full_list.js +0 -216
  129. data/docs/js/jquery.js +0 -4
  130. data/docs/method_list.html +0 -4075
  131. data/docs/top-level-namespace.html +0 -126
  132. data/lib/puppeteer/async_await_behavior.rb +0 -38
@@ -1,5 +1,8 @@
1
1
  # utility methods for Concurrent::Promises.
2
2
  module Puppeteer::ConcurrentRubyUtils
3
+ # wait for all promises.
4
+ # REMARK: This method doesn't assure the order of calling.
5
+ # for example, await_all(async1, async2) calls calls2 -> calls1 often.
3
6
  def await_all(*args)
4
7
  if args.length == 1 && args[0].is_a?(Enumerable)
5
8
  Concurrent::Promises.zip(*(args[0])).value!
@@ -8,6 +11,9 @@ module Puppeteer::ConcurrentRubyUtils
8
11
  end
9
12
  end
10
13
 
14
+ # wait for first promises.
15
+ # REMARK: This method doesn't assure the order of calling.
16
+ # for example, await_all(async1, async2) calls calls2 -> calls1 often.
11
17
  def await_any(*args)
12
18
  if args.length == 1 && args[0].is_a?(Enumerable)
13
19
  Concurrent::Promises.any(*(args[0])).value!
@@ -29,8 +35,12 @@ module Puppeteer::ConcurrentRubyUtils
29
35
  Concurrent::Promises.future(&block)
30
36
  end
31
37
 
32
- def resolvable_future
33
- Concurrent::Promises.resolvable_future
38
+ def resolvable_future(&block)
39
+ future = Concurrent::Promises.resolvable_future
40
+ if block
41
+ block.call(future)
42
+ end
43
+ future
34
44
  end
35
45
  end
36
46
 
@@ -3,7 +3,7 @@ require 'json'
3
3
  class Puppeteer::Connection
4
4
  include Puppeteer::DebugPrint
5
5
  include Puppeteer::EventCallbackable
6
- using Puppeteer::AsyncAwaitBehavior
6
+ using Puppeteer::DefineAsyncMethod
7
7
 
8
8
  class ProtocolError < StandardError
9
9
  def initialize(method:, error_message:, error_data: nil)
@@ -44,7 +44,9 @@ class Puppeteer::Connection
44
44
 
45
45
  @transport = transport
46
46
  @transport.on_message do |data|
47
- async_handle_message(JSON.parse(data))
47
+ message = JSON.parse(data)
48
+ sleep_before_handling_message(message)
49
+ async_handle_message(message)
48
50
  end
49
51
  @transport.on_close do |reason, code|
50
52
  handle_close(reason, code)
@@ -54,6 +56,16 @@ class Puppeteer::Connection
54
56
  @closed = false
55
57
  end
56
58
 
59
+ private def sleep_before_handling_message(message)
60
+ # Puppeteer doesn't handle any Network monitoring responses.
61
+ # So we don't have to sleep.
62
+ return if message['method']&.start_with?('Network.')
63
+
64
+ # For some reasons, sleeping a bit reduces trivial errors...
65
+ # 4ms is an interval of internal shared timer of WebKit.
66
+ sleep 0.004
67
+ end
68
+
57
69
  def self.from_session(session)
58
70
  session.connection
59
71
  end
@@ -198,9 +210,7 @@ class Puppeteer::Connection
198
210
  end
199
211
  end
200
212
 
201
- private async def async_handle_message(message)
202
- handle_message(message)
203
- end
213
+ private define_async_method :async_handle_message
204
214
 
205
215
  private def handle_close
206
216
  return if @closed
@@ -239,16 +249,6 @@ class Puppeteer::Connection
239
249
  def create_session(target_info)
240
250
  result = send_message('Target.attachToTarget', targetId: target_info.target_id, flatten: true)
241
251
  session_id = result['sessionId']
242
-
243
- # Target.attachedToTarget is often notified after the result of Target.attachToTarget.
244
- # D, [2020-04-04T23:04:30.736311 #91875] DEBUG -- : RECV << {"id"=>2, "result"=>{"sessionId"=>"DA002F8A95B04710502CB40D8430B95A"}}
245
- # D, [2020-04-04T23:04:30.736649 #91875] DEBUG -- : RECV << {"method"=>"Target.attachedToTarget", "params"=>{"sessionId"=>"DA002F8A95B04710502CB40D8430B95A", "targetInfo"=>{"targetId"=>"EBAB949A7DE63F12CB94268AD3A9976B", "type"=>"page", "title"=>"about:blank", "url"=>"about:blank", "attached"=>true, "browserContextId"=>"46D23767E9B79DD9E589101121F6DADD"}, "waitingForDebugger"=>false}}
246
- # So we have to wait for "Target.attachedToTarget" a bit.
247
- 20.times do
248
- if @sessions[session_id]
249
- return @sessions[session_id]
250
- end
251
- sleep 0.1
252
- end
252
+ @sessions[session_id]
253
253
  end
254
254
  end
@@ -0,0 +1,23 @@
1
+ module Puppeteer::DefineAsyncMethod
2
+ refine Class do
3
+ def define_async_method(async_method_name)
4
+ unless async_method_name.to_s.start_with?('async_')
5
+ raise ArgumentError.new('async method name should start with "async_"')
6
+ end
7
+
8
+ if method_defined?(async_method_name) || private_method_defined?(async_method_name)
9
+ raise ArgumentError.new("#{async_method_name} is already defined")
10
+ end
11
+
12
+ original_method = instance_method(async_method_name[6..-1])
13
+ define_method(async_method_name) do |*args|
14
+ Concurrent::Promises.future do
15
+ original_method.bind(self).call(*args)
16
+ rescue => err
17
+ Logger.new(STDERR).warn(err)
18
+ raise err
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -2,7 +2,7 @@ require 'thread'
2
2
 
3
3
  # https://github.com/puppeteer/puppeteer/blob/master/src/DOMWorld.js
4
4
  class Puppeteer::DOMWorld
5
- using Puppeteer::AsyncAwaitBehavior
5
+ using Puppeteer::DefineAsyncMethod
6
6
 
7
7
  # @param {!Puppeteer.FrameManager} frameManager
8
8
  # @param {!Puppeteer.Frame} frame
@@ -12,7 +12,6 @@ class Puppeteer::DOMWorld
12
12
  @frame = frame
13
13
  @timeout_settings = timeout_settings
14
14
  @context_promise = resolvable_future
15
- @pending_destroy = []
16
15
  @wait_tasks = Set.new
17
16
  @detached = false
18
17
  end
@@ -24,22 +23,12 @@ class Puppeteer::DOMWorld
24
23
  @wait_tasks
25
24
  end
26
25
 
27
- # @param {?Puppeteer.ExecutionContext} context
26
+ # @param context [Puppeteer::ExecutionContext]
28
27
  def context=(context)
29
- # D, [2020-04-12T22:45:03.938754 #46154] DEBUG -- : RECV << {"method"=>"Runtime.executionContextCreated", "params"=>{"context"=>{"id"=>3, "origin"=>"https://github.com", "name"=>"", "auxData"=>{"isDefault"=>true, "type"=>"default", "frameId"=>"3AD7F1E82BCBA88BFE31D03BC49FF6CB"}}}, "sessionId"=>"636CEF0C4FEAFC4FE815E9E7B5F7BA68"}
30
- # D, [2020-04-12T22:45:03.938856 #46154] DEBUG -- : RECV << {"method"=>"Runtime.executionContextCreated", "params"=>{"context"=>{"id"=>4, "origin"=>"://", "name"=>"__puppeteer_utility_world__", "auxData"=>{"isDefault"=>false, "type"=>"isolated", "frameId"=>"3AD7F1E82BCBA88BFE31D03BC49FF6CB"}}}, "sessionId"=>"636CEF0C4FEAFC4FE815E9E7B5F7BA68"}
31
- # D, [2020-04-12T22:45:03.938960 #46154] DEBUG -- : RECV << {"method"=>"Runtime.executionContextDestroyed", "params"=>{"executionContextId"=>1}, "sessionId"=>"636CEF0C4FEAFC4FE815E9E7B5F7BA68"}
32
- # D, [2020-04-12T22:45:03.939110 #46154] DEBUG -- : RECV << {"method"=>"Page.frameNavigated", "params"=>{"frame"=>{"id"=>"3AD7F1E82BCBA88BFE31D03BC49FF6CB", "loaderId"=>"301B349884E582986C502CBE020966DF", "url"=>"https://github.com/", "securityOrigin"=>"https://github.com", "mimeType"=>"text/html"}}, "sessionId"=>"636CEF0C4FEAFC4FE815E9E7B5F7BA68"}
33
- # D, [2020-04-12T22:45:03.939793 #46154] DEBUG -- : RECV << {"method"=>"Runtime.executionContextDestroyed", "params"=>{"executionContextId"=>2}, "sessionId"=>"636CEF0C4FEAFC4FE815E9E7B5F7BA68"}
34
- # executionContextDestroyed is often notified after executionContextCreated.
35
-
36
28
  if context
37
- if @context_promise.fulfilled?
38
- @pending_destroy << context._context_id
39
- @document = nil
40
- @context_promise = resolvable_future
29
+ unless @context_promise.resolved?
30
+ @context_promise.fulfill(context)
41
31
  end
42
- @context_promise.fulfill(context)
43
32
  @wait_tasks.each(&:async_rerun)
44
33
  else
45
34
  raise ArgumentError.new("context should now be nil. Use #delete_context for clearing document.")
@@ -47,12 +36,8 @@ class Puppeteer::DOMWorld
47
36
  end
48
37
 
49
38
  def delete_context(execution_context_id)
50
- if @pending_destroy.include?(execution_context_id)
51
- @pending_destroy.delete(execution_context_id)
52
- else
53
- @document = nil
54
- @context_promise = resolvable_future
55
- end
39
+ @document = nil
40
+ @context_promise = resolvable_future
56
41
  end
57
42
 
58
43
  def has_context?
@@ -97,17 +82,18 @@ class Puppeteer::DOMWorld
97
82
  document.S(selector)
98
83
  end
99
84
 
100
- class DocumentEvaluationError < StandardError; end
101
-
102
85
  private def evaluate_document
103
86
  # sometimes execution_context.evaluate_handle('document') returns null object.
104
87
  # D, [2020-04-24T02:17:51.023631 #220] DEBUG -- : RECV << {"id"=>20, "result"=>{"result"=>{"type"=>"object", "subtype"=>"null", "value"=>nil}}, "sessionId"=>"78E9CF1E14D81294E320E7C20E5CDE06"}
105
88
  # retry if so.
106
- 5.times do
107
- handle = execution_context.evaluate_handle('document')
108
- return handle if handle.is_a?(Puppeteer::ElementHandle)
89
+ Timeout.timeout(3) do
90
+ loop do
91
+ handle = execution_context.evaluate_handle('document')
92
+ return handle if handle.is_a?(Puppeteer::ElementHandle)
93
+ end
109
94
  end
110
- raise DocumentEvaluationError.new("'document' object cannot be evaluated as an Element")
95
+ rescue Timeout::Error
96
+ raise 'Bug of puppeteer-ruby...'
111
97
  end
112
98
 
113
99
  private def document
@@ -146,45 +132,48 @@ class Puppeteer::DOMWorld
146
132
  document.SS(selector)
147
133
  end
148
134
 
149
- # /**
150
- # * @return {!Promise<String>}
151
- # */
152
- # async content() {
153
- # return await this.evaluate(() => {
154
- # let retVal = '';
155
- # if (document.doctype)
156
- # retVal = new XMLSerializer().serializeToString(document.doctype);
157
- # if (document.documentElement)
158
- # retVal += document.documentElement.outerHTML;
159
- # return retVal;
160
- # });
161
- # }
135
+ # @return [String]
136
+ def content
137
+ evaluate <<-JAVASCRIPT
138
+ () => {
139
+ let retVal = '';
140
+ if (document.doctype)
141
+ retVal = new XMLSerializer().serializeToString(document.doctype);
142
+ if (document.documentElement)
143
+ retVal += document.documentElement.outerHTML;
144
+ return retVal;
145
+ }
146
+ JAVASCRIPT
147
+ end
162
148
 
163
- # /**
164
- # * @param {string} html
165
- # * @param {!{timeout?: number, waitUntil?: string|!Array<string>}=} options
166
- # */
167
- # async setContent(html, options = {}) {
168
- # const {
169
- # waitUntil = ['load'],
170
- # timeout = this._timeoutSettings.navigationTimeout(),
171
- # } = options;
172
- # // We rely upon the fact that document.open() will reset frame lifecycle with "init"
173
- # // lifecycle event. @see https://crrev.com/608658
174
- # await this.evaluate(html => {
175
- # document.open();
176
- # document.write(html);
177
- # document.close();
178
- # }, html);
179
- # const watcher = new LifecycleWatcher(this._frameManager, this._frame, waitUntil, timeout);
180
- # const error = await Promise.race([
181
- # watcher.timeoutOrTerminationPromise(),
182
- # watcher.lifecyclePromise(),
183
- # ]);
184
- # watcher.dispose();
185
- # if (error)
186
- # throw error;
187
- # }
149
+ # @param html [String]
150
+ # @param timeout [Integer]
151
+ # @param wait_until [String|Array<String>]
152
+ def set_content(html, timeout: nil, wait_until: nil)
153
+ option_wait_until = [wait_until || 'load'].flatten
154
+ option_timeout = @timeout_settings.navigation_timeout
155
+
156
+ # We rely upon the fact that document.open() will reset frame lifecycle with "init"
157
+ # lifecycle event. @see https://crrev.com/608658
158
+ js = <<-JAVASCRIPT
159
+ (html) => {
160
+ document.open();
161
+ document.write(html);
162
+ document.close();
163
+ }
164
+ JAVASCRIPT
165
+ evaluate(js, html)
166
+
167
+ watcher = Puppeteer::LifecycleWatcher.new(@frame_manager, @frame, option_wait_until, option_timeout)
168
+ begin
169
+ await_any(
170
+ watcher.timeout_or_termination_promise,
171
+ watcher.lifecycle_promise,
172
+ )
173
+ ensure
174
+ watcher.dispose
175
+ end
176
+ end
188
177
 
189
178
  # /**
190
179
  # * @param {!{url?: string, path?: string, content?: string, type?: string}} options
@@ -410,12 +399,10 @@ class Puppeteer::DOMWorld
410
399
  # return new WaitTask(this, pageFunction, 'function', polling, timeout, ...args).promise;
411
400
  # }
412
401
 
413
- # /**
414
- # * @return {!Promise<string>}
415
- # */
416
- # async title() {
417
- # return this.evaluate(() => document.title);
418
- # }
402
+ # @return [String]
403
+ def title
404
+ evaluate('() => document.title')
405
+ end
419
406
 
420
407
  # @param selector_or_xpath [String]
421
408
  # @param is_xpath [Boolean]
@@ -4,7 +4,7 @@ require_relative './element_handle/point'
4
4
 
5
5
  class Puppeteer::ElementHandle < Puppeteer::JSHandle
6
6
  include Puppeteer::IfPresent
7
- using Puppeteer::AsyncAwaitBehavior
7
+ using Puppeteer::DefineAsyncMethod
8
8
 
9
9
  # @param context [Puppeteer::ExecutionContext]
10
10
  # @param client [Puppeteer::CDPSession]
@@ -105,11 +105,11 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
105
105
  end
106
106
  end
107
107
 
108
- # async hover() {
109
- # await this._scrollIntoViewIfNeeded();
110
- # const {x, y} = await this._clickablePoint();
111
- # await this._page.mouse.move(x, y);
112
- # }
108
+ def hover
109
+ scroll_into_view_if_needed
110
+ point = clickable_point
111
+ @page.mouse.move(point.x, point.y)
112
+ end
113
113
 
114
114
  # @param delay [Number]
115
115
  # @param button [String] "left"|"right"|"middle"
@@ -120,12 +120,7 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
120
120
  @page.mouse.click(point.x, point.y, delay: delay, button: button, click_count: click_count)
121
121
  end
122
122
 
123
- # @param delay [Number]
124
- # @param button [String] "left"|"right"|"middle"
125
- # @param click_count [Number]
126
- async def async_click(delay: nil, button: nil, click_count: nil)
127
- click(delay: delay, button: button, click_count: click_count)
128
- end
123
+ define_async_method :async_click
129
124
 
130
125
  # @return [Array<String>]
131
126
  def select(*values)
@@ -195,17 +190,13 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
195
190
  @page.touchscreen.tap(point.x, point.y)
196
191
  end
197
192
 
198
- async def async_tap
199
- tap
200
- end
193
+ define_async_method :async_tap
201
194
 
202
195
  def focus
203
196
  evaluate('element => element.focus()')
204
197
  end
205
198
 
206
- async def async_focus
207
- focus
208
- end
199
+ define_async_method :async_focus
209
200
 
210
201
  # @param text [String]
211
202
  # @param delay [number|nil]
@@ -214,12 +205,7 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
214
205
  @page.keyboard.type_text(text, delay: delay)
215
206
  end
216
207
 
217
- # @param text [String]
218
- # @param delay [number|nil]
219
- # @return [Future]
220
- async def async_type_text(text, delay: nil)
221
- type_text(text, delay: delay)
222
- end
208
+ define_async_method :async_type_text
223
209
 
224
210
  # @param key [String]
225
211
  # @param delay [number|nil]
@@ -228,12 +214,7 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
228
214
  @page.keyboard.press(key, delay: delay)
229
215
  end
230
216
 
231
- # @param key [String]
232
- # @param delay [number|nil]
233
- # @return [Future]
234
- async def async_press(key, delay: nil)
235
- press(key, delay: delay)
236
- end
217
+ define_async_method :async_press
237
218
 
238
219
  # @return [BoundingBox|nil]
239
220
  def bounding_box
@@ -356,13 +337,7 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
356
337
  result
357
338
  end
358
339
 
359
- # `$eval()` in JavaScript. $ is not allowed to use as a method name in Ruby.
360
- # @param selector [String]
361
- # @param page_function [String]
362
- # @return [Object]
363
- async def async_Seval(selector, page_function, *args)
364
- Seval(selector, page_function, *args)
365
- end
340
+ define_async_method :async_Seval
366
341
 
367
342
  # `$$eval()` in JavaScript. $ is not allowed to use as a method name in Ruby.
368
343
  # @param selector [String]
@@ -379,13 +354,7 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
379
354
  result
380
355
  end
381
356
 
382
- # `$$eval()` in JavaScript. $ is not allowed to use as a method name in Ruby.
383
- # @param selector [String]
384
- # @param page_function [String]
385
- # @return [Object]
386
- async def async_SSeval(selector, page_function, *args)
387
- SSeval(selector, page_function, *args)
388
- end
357
+ define_async_method :async_SSeval
389
358
 
390
359
  # `$x()` in JavaScript. $ is not allowed to use as a method name in Ruby.
391
360
  # @param expression [String]
@@ -408,6 +377,8 @@ class Puppeteer::ElementHandle < Puppeteer::JSHandle
408
377
  properties.values.map(&:as_element).compact
409
378
  end
410
379
 
380
+ define_async_method :async_Sx
381
+
411
382
  # /**
412
383
  # * @returns {!Promise<boolean>}
413
384
  # */
@@ -1,5 +1,5 @@
1
1
  class Puppeteer::EmulationManager
2
- using Puppeteer::AsyncAwaitBehavior
2
+ using Puppeteer::DefineAsyncMethod
3
3
 
4
4
  # @param {!Puppeteer.CDPSession} client
5
5
  def initialize(client)
@@ -38,9 +38,5 @@ class Puppeteer::EmulationManager
38
38
  reload_needed
39
39
  end
40
40
 
41
- # @param viewport [Puppeteer::Viewport]
42
- # @return [Future<true|false>]
43
- async def async_emulate_viewport(viewport)
44
- emulate_viewport(viewport)
45
- end
41
+ define_async_method :async_emulate_viewport
46
42
  end
@@ -39,6 +39,17 @@ module Puppeteer::EventCallbackable
39
39
  end
40
40
  end
41
41
 
42
+ def observe_first(event_name, &block)
43
+ listener_id = add_event_listener(event_name) do |*args, **kwargs|
44
+ if kwargs.empty?
45
+ block.call(*args)
46
+ else
47
+ block.call(*args, **kwargs)
48
+ end
49
+ remove_event_listener(listener_id)
50
+ end
51
+ end
52
+
42
53
  def on_event(event_name, &block)
43
54
  @event_callbackable_handlers ||= {}
44
55
  @event_callbackable_handlers[event_name] = block