Class: Capybara::Playwright::Browser

Inherits:
Object
  • Object
show all
Extended by:
Forwardable
Includes:
TmpdirOwner
Defined in:
lib/capybara/playwright/browser.rb

Overview

Responsibility of this class is:

  • Handling Capybara::Driver commands.

  • Managing Playwright browser contexts and pages.

Note that this class doesn’t manage Playwright::Browser. We should not use Playwright::Browser#close in this class.

Defined Under Namespace

Classes: NoSuchWindowError

Instance Method Summary collapse

Methods included from TmpdirOwner

#remove_tmpdir, #tmpdir

Constructor Details

#initialize(driver:, internal_logger:, playwright_browser:, page_options:, record_video: false, callback_on_save_trace: nil, default_timeout: nil, default_navigation_timeout: nil) ⇒ Browser

Returns a new instance of Browser.



18
19
20
21
22
23
24
25
26
27
28
29
30
# File 'lib/capybara/playwright/browser.rb', line 18

def initialize(driver:, internal_logger:, playwright_browser:, page_options:, record_video: false, callback_on_save_trace: nil, default_timeout: nil, default_navigation_timeout: nil)
  @driver = driver
  @internal_logger = internal_logger
  @playwright_browser = playwright_browser
  @page_options = page_options
  if record_video
    @page_options[:record_video_dir] ||= tmpdir
  end
  @callback_on_save_trace = callback_on_save_trace
  @default_timeout = default_timeout
  @default_navigation_timeout = default_navigation_timeout
  @playwright_page = create_page(create_browser_context)
end

Instance Method Details

#accept_modal(dialog_type, **options, &block) ⇒ Object



347
348
349
350
351
# File 'lib/capybara/playwright/browser.rb', line 347

def accept_modal(dialog_type, **options, &block)
  assert_page_alive {
    @playwright_page.capybara_accept_modal(dialog_type, **options, &block)
  }
end

#active_elementObject



187
188
189
190
191
192
193
194
# File 'lib/capybara/playwright/browser.rb', line 187

def active_element
  el = @playwright_page.capybara_current_frame.evaluate_handle('() => document.activeElement')
  if el
    Node.new(@driver, @internal_logger, @playwright_page, el)
  else
    nil
  end
end

#clear_browser_contextsObject



58
59
60
61
62
63
64
65
66
67
68
# File 'lib/capybara/playwright/browser.rb', line 58

def clear_browser_contexts
  if @callback_on_save_trace
    @playwright_browser.contexts.each do |browser_context|
      filename = SecureRandom.hex(8)
      zip_path = File.join(tmpdir, "#{filename}.zip")
      browser_context.tracing.stop(path: zip_path)
      @callback_on_save_trace.call(zip_path)
    end
  end
  @playwright_browser.contexts.each(&:close)
end

#close_window(handle) ⇒ Object



306
307
308
309
310
311
312
313
314
# File 'lib/capybara/playwright/browser.rb', line 306

def close_window(handle)
  on_window(handle) do |page|
    page.close

    if @playwright_page&.guid == handle
      @playwright_page = nil
    end
  end
end

#current_urlObject



70
71
72
73
74
# File 'lib/capybara/playwright/browser.rb', line 70

def current_url
  assert_page_alive {
    @playwright_page.url
  }
end

#current_window_handleObject



274
275
276
# File 'lib/capybara/playwright/browser.rb', line 274

def current_window_handle
  @playwright_page&.guid
end

#dismiss_modal(dialog_type, **options, &block) ⇒ Object



353
354
355
356
357
# File 'lib/capybara/playwright/browser.rb', line 353

def dismiss_modal(dialog_type, **options, &block)
  assert_page_alive {
    @playwright_page.capybara_dismiss_modal(dialog_type, **options, &block)
  }
end

#evaluate_async_script(script, *args) ⇒ Object



171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
# File 'lib/capybara/playwright/browser.rb', line 171

def evaluate_async_script(script, *args)
  assert_page_alive {
    js = <<~JAVASCRIPT
    function(_arguments){
      let args = Array.prototype.slice.call(_arguments);
      return new Promise((resolve, reject) => {
        args.push(resolve);
        (function(){ #{script} }).apply(this, args);
      });
    }
    JAVASCRIPT
    result = @playwright_page.capybara_current_frame.evaluate_handle(js, arg: unwrap_node(args))
    wrap_node(result)
  }
end

#evaluate_script(script, *args) ⇒ Object



164
165
166
167
168
169
# File 'lib/capybara/playwright/browser.rb', line 164

def evaluate_script(script, *args)
  assert_page_alive {
    result = @playwright_page.capybara_current_frame.evaluate_handle("function (arguments) { return #{script} }", arg: unwrap_node(args))
    wrap_node(result)
  }
end

#execute_script(script, *args) ⇒ Object



157
158
159
160
161
162
# File 'lib/capybara/playwright/browser.rb', line 157

def execute_script(script, *args)
  assert_page_alive {
    @playwright_page.capybara_current_frame.evaluate("function (arguments) { #{script} }", arg: unwrap_node(args))
  }
  nil
end

#find_css(query, **options) ⇒ Object



105
106
107
108
109
110
111
# File 'lib/capybara/playwright/browser.rb', line 105

def find_css(query, **options)
  assert_page_alive {
    @playwright_page.capybara_current_frame.query_selector_all(query).map do |el|
      Node.new(@driver, @internal_logger, @playwright_page, el)
    end
  }
end

#find_xpath(query, **options) ⇒ Object



97
98
99
100
101
102
103
# File 'lib/capybara/playwright/browser.rb', line 97

def find_xpath(query, **options)
  assert_page_alive {
    @playwright_page.capybara_current_frame.query_selector_all("xpath=#{query}").map do |el|
      Node.new(@driver, @internal_logger, @playwright_page, el)
    end
  }
end

#fullscreen_window(handle) ⇒ Object



338
339
340
341
342
343
344
345
# File 'lib/capybara/playwright/browser.rb', line 338

def fullscreen_window(handle)
  @internal_logger.warn("fullscreen_window is not supported in Playwright driver")
  # incomplete in Playwright
  # ref: https://github.com/twalpole/apparition/blob/11aca464b38b77585191b7e302be2e062bdd369d/lib/capybara/apparition/page.rb#L341
  on_window(handle) do |page|
    page.evaluate('() => document.body.requestFullscreen()')
  end
end

#go_backObject



145
146
147
148
149
# File 'lib/capybara/playwright/browser.rb', line 145

def go_back
  assert_page_alive {
    @playwright_page.go_back
  }
end

#go_forwardObject



151
152
153
154
155
# File 'lib/capybara/playwright/browser.rb', line 151

def go_forward
  assert_page_alive {
    @playwright_page.go_forward
  }
end

#htmlObject



125
126
127
128
129
130
131
132
133
134
135
136
137
# File 'lib/capybara/playwright/browser.rb', line 125

def html
  assert_page_alive {
    js = <<~JAVASCRIPT
    () => {
      let html = '';
      if (document.doctype) html += new XMLSerializer().serializeToString(document.doctype);
      if (document.documentElement) html += document.documentElement.outerHTML;
      return html;
    }
    JAVASCRIPT
    @playwright_page.capybara_current_frame.evaluate(js)
  }
end

#maximize_window(handle) ⇒ Object



328
329
330
331
332
333
334
335
336
# File 'lib/capybara/playwright/browser.rb', line 328

def maximize_window(handle)
  @internal_logger.warn("maximize_window is not supported in Playwright driver")
  # incomplete in Playwright
  # ref: https://github.com/twalpole/apparition/blob/11aca464b38b77585191b7e302be2e062bdd369d/lib/capybara/apparition/page.rb#L346
  on_window(handle) do |page|
    screen_size = page.evaluate('() => ({ width: window.screen.width, height: window.screen.height})')
    page.viewport_size = screen_size
  end
end

#open_new_window(kind = :tab) ⇒ Object



278
279
280
281
282
283
284
285
286
287
# File 'lib/capybara/playwright/browser.rb', line 278

def open_new_window(kind = :tab)
  browser_context =
    if kind == :tab
      @playwright_page&.context || create_browser_context
    else
      create_browser_context
    end

  create_page(browser_context)
end

#raw_screenshot(**options) ⇒ Object

Not used by Capybara::Session. Intended to be directly called by user.



206
207
208
209
210
# File 'lib/capybara/playwright/browser.rb', line 206

def raw_screenshot(**options)
  return nil if !@playwright_page || @playwright_page.closed?

  @playwright_page.screenshot(**options)
end

#refreshObject



91
92
93
94
95
# File 'lib/capybara/playwright/browser.rb', line 91

def refresh
  assert_page_alive {
    @playwright_page.capybara_current_frame.evaluate('() => { location.reload(true) }')
  }
end

#resize_window_to(handle, width, height) ⇒ Object



322
323
324
325
326
# File 'lib/capybara/playwright/browser.rb', line 322

def resize_window_to(handle, width, height)
  on_window(handle) do |page|
    page.viewport_size = { width: width, height: height }
  end
end

#response_headersObject



113
114
115
116
117
# File 'lib/capybara/playwright/browser.rb', line 113

def response_headers
  assert_page_alive {
    @playwright_page.capybara_response_headers
  }
end

#save_screenshot(path, **options) ⇒ Object



212
213
214
215
216
# File 'lib/capybara/playwright/browser.rb', line 212

def save_screenshot(path, **options)
  assert_page_alive {
    @playwright_page.screenshot(path: path)
  }
end

#send_keys(*args) ⇒ Object



218
219
220
# File 'lib/capybara/playwright/browser.rb', line 218

def send_keys(*args)
  Node::SendKeys.new(@playwright_page.keyboard, args).execute
end

#status_codeObject



119
120
121
122
123
# File 'lib/capybara/playwright/browser.rb', line 119

def status_code
  assert_page_alive {
    @playwright_page.capybara_status_code
  }
end

#switch_to_frame(frame) ⇒ Object



222
223
224
225
226
227
228
229
230
231
232
233
234
235
# File 'lib/capybara/playwright/browser.rb', line 222

def switch_to_frame(frame)
  assert_page_alive {
    case frame
    when :top
      @playwright_page.capybara_reset_frames
    when :parent
      @playwright_page.capybara_pop_frame
    else
      playwright_frame = frame.native.content_frame
      raise ArgumentError.new("Not a frame element: #{frame}") unless playwright_frame
      @playwright_page.capybara_push_frame(playwright_frame)
    end
  }
end

#switch_to_window(handle) ⇒ Object



298
299
300
301
302
303
304
# File 'lib/capybara/playwright/browser.rb', line 298

def switch_to_window(handle)
  if @playwright_page&.guid != handle
    on_window(handle) do |page|
      @playwright_page = page.tap(&:bring_to_front)
    end
  end
end

#titleObject



139
140
141
142
143
# File 'lib/capybara/playwright/browser.rb', line 139

def title
  assert_page_alive {
    @playwright_page.title
  }
end

#video_pathObject

Not used by Capybara::Session. Intended to be directly called by user.



198
199
200
201
202
# File 'lib/capybara/playwright/browser.rb', line 198

def video_path
  return nil if !@playwright_page || @playwright_page.closed?

  @playwright_page.video&.path
end

#visit(path) ⇒ Object



76
77
78
79
80
81
82
83
84
85
86
87
88
89
# File 'lib/capybara/playwright/browser.rb', line 76

def visit(path)
  assert_page_alive {
    url =
    if Capybara.app_host
      Addressable::URI.parse(Capybara.app_host) + path
    elsif Capybara.default_host
      Addressable::URI.parse(Capybara.default_host) + path
    else
      path
    end

    @playwright_page.capybara_current_frame.goto(url)
  }
end

#window_handlesObject



270
271
272
# File 'lib/capybara/playwright/browser.rb', line 270

def window_handles
  pages.map(&:guid)
end

#window_size(handle) ⇒ Object



316
317
318
319
320
# File 'lib/capybara/playwright/browser.rb', line 316

def window_size(handle)
  on_window(handle) do |page|
    page.evaluate('() => [window.innerWidth, window.innerHeight]')
  end
end

#with_playwright_page(&block) ⇒ Object



405
406
407
408
409
# File 'lib/capybara/playwright/browser.rb', line 405

def with_playwright_page(&block)
  assert_page_alive {
    block.call(@playwright_page)
  }
end