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:, 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
# File 'lib/capybara/playwright/browser.rb', line 18

def initialize(driver:, playwright_browser:, page_options:, record_video: false, callback_on_save_trace: nil, default_timeout: nil, default_navigation_timeout: nil)
  @driver = driver
  @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



345
346
347
348
349
# File 'lib/capybara/playwright/browser.rb', line 345

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

#active_elementObject



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

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

#clear_browser_contextsObject



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

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



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

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

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

#current_urlObject



68
69
70
71
72
# File 'lib/capybara/playwright/browser.rb', line 68

def current_url
  assert_page_alive {
    @playwright_page.url
  }
end

#current_window_handleObject



272
273
274
# File 'lib/capybara/playwright/browser.rb', line 272

def current_window_handle
  @playwright_page&.guid
end

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



351
352
353
354
355
# File 'lib/capybara/playwright/browser.rb', line 351

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



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

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



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

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



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

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



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

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

#find_xpath(query, **options) ⇒ Object



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

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

#fullscreen_window(handle) ⇒ Object



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

def fullscreen_window(handle)
  puts "[WARNING] 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



143
144
145
146
147
# File 'lib/capybara/playwright/browser.rb', line 143

def go_back
  assert_page_alive {
    @playwright_page.go_back
  }
end

#go_forwardObject



149
150
151
152
153
# File 'lib/capybara/playwright/browser.rb', line 149

def go_forward
  assert_page_alive {
    @playwright_page.go_forward
  }
end

#htmlObject



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

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



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

def maximize_window(handle)
  puts "[WARNING] 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



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

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.



204
205
206
207
208
# File 'lib/capybara/playwright/browser.rb', line 204

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

  @playwright_page.screenshot(**options)
end

#refreshObject



89
90
91
92
93
# File 'lib/capybara/playwright/browser.rb', line 89

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

#resize_window_to(handle, width, height) ⇒ Object



320
321
322
323
324
# File 'lib/capybara/playwright/browser.rb', line 320

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

#response_headersObject



111
112
113
114
115
# File 'lib/capybara/playwright/browser.rb', line 111

def response_headers
  assert_page_alive {
    @playwright_page.capybara_response_headers
  }
end

#save_screenshot(path, **options) ⇒ Object



210
211
212
213
214
# File 'lib/capybara/playwright/browser.rb', line 210

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

#send_keys(*args) ⇒ Object



216
217
218
# File 'lib/capybara/playwright/browser.rb', line 216

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

#status_codeObject



117
118
119
120
121
# File 'lib/capybara/playwright/browser.rb', line 117

def status_code
  assert_page_alive {
    @playwright_page.capybara_status_code
  }
end

#switch_to_frame(frame) ⇒ Object



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

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



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

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



137
138
139
140
141
# File 'lib/capybara/playwright/browser.rb', line 137

def title
  assert_page_alive {
    @playwright_page.title
  }
end

#video_pathObject

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



196
197
198
199
200
# File 'lib/capybara/playwright/browser.rb', line 196

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

  @playwright_page.video&.path
end

#visit(path) ⇒ Object



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

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



268
269
270
# File 'lib/capybara/playwright/browser.rb', line 268

def window_handles
  pages.map(&:guid)
end

#window_size(handle) ⇒ Object



314
315
316
317
318
# File 'lib/capybara/playwright/browser.rb', line 314

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

#with_playwright_page(&block) ⇒ Object



403
404
405
406
407
# File 'lib/capybara/playwright/browser.rb', line 403

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