class Poltergeist.Browser constructor: (@owner, width, height) -> @width = width || 1024 @height = height || 768 @state = 'default' @page_stack = [] @page_id = 0 @js_errors = true @_debug = false this.resetPage() resetPage: -> if @page? @page.release() phantom.clearCookies() @page = new Poltergeist.WebPage @page.setViewportSize(width: @width, height: @height) @page.onLoadStarted = => this.setState 'loading' if @state == 'mouse_event' @page.onNavigationRequested = (url, navigation) => this.setState 'loading' if @state == 'mouse_event' && navigation == 'FormSubmitted' @page.onLoadFinished = (status) => if @state == 'loading' this.sendResponse(status: status, position: @last_mouse_event) this.setState 'default' else if @state == 'awaiting_frame_load' this.sendResponse(true) this.setState 'default' @page.onInitialized = => @page_id += 1 @page.onPageCreated = (sub_page) => if @state == 'awaiting_sub_page' name = @page_name @page_name = null this.setState 'default' # At this point subpage isn't fully initialized, so we can't check # its name. Instead, we just schedule another attempt to push the # window. setTimeout((=> this.push_window(name)), 0) runCommand: (name, args) -> this.setState "default" this[name].apply(this, args) debug: (message) -> if @_debug console.log "poltergeist [#{new Date().getTime()}] #{message}" setState: (state) -> return if @state == state this.debug "state #{@state} -> #{state}" @state = state sendResponse: (response) -> errors = @page.errors() @page.clearErrors() if errors.length > 0 && @js_errors @owner.sendError(new Poltergeist.JavascriptError(errors)) else @owner.sendResponse(response) add_extension: (extension) -> @page.injectExtension extension this.sendResponse 'success' node: (page_id, id) -> if page_id == @page_id @page.get(id) else throw new Poltergeist.ObsoleteNode visit: (url) -> this.setState 'loading' prev_url = @page.currentUrl() @page.open(url) if /#/.test(url) && prev_url.split('#')[0] == url.split('#')[0] # hashchange occurred, so there will be no onLoadFinished this.setState 'default' this.sendResponse 'success' current_url: -> this.sendResponse @page.currentUrl() status_code: -> this.sendResponse @page.statusCode() body: -> this.sendResponse @page.content() source: -> this.sendResponse @page.source() title: -> this.sendResponse @page.title() find: (method, selector) -> this.sendResponse(page_id: @page_id, ids: @page.find(method, selector)) find_within: (page_id, id, method, selector) -> this.sendResponse this.node(page_id, id).find(method, selector) all_text: (page_id, id) -> this.sendResponse this.node(page_id, id).allText() visible_text: (page_id, id) -> this.sendResponse this.node(page_id, id).visibleText() attribute: (page_id, id, name) -> this.sendResponse this.node(page_id, id).getAttribute(name) value: (page_id, id) -> this.sendResponse this.node(page_id, id).value() set: (page_id, id, value) -> this.node(page_id, id).set(value) this.sendResponse(true) # PhantomJS only allows us to reference the element by CSS selector, not XPath, # so we have to add an attribute to the element to identify it, then remove it # afterwards. select_file: (page_id, id, value) -> node = this.node(page_id, id) @page.beforeUpload(node.id) @page.uploadFile('[_poltergeist_selected]', value) @page.afterUpload(node.id) this.sendResponse(true) select: (page_id, id, value) -> this.sendResponse this.node(page_id, id).select(value) tag_name: (page_id, id) -> this.sendResponse this.node(page_id, id).tagName() visible: (page_id, id) -> this.sendResponse this.node(page_id, id).isVisible() disabled: (page_id, id) -> this.sendResponse this.node(page_id, id).isDisabled() evaluate: (script) -> this.sendResponse @page.evaluate("function() { return #{script} }") execute: (script) -> @page.execute("function() { #{script} }") this.sendResponse(true) push_frame: (name, timeout = new Date().getTime() + 2000) -> if @page.pushFrame(name) if @page.currentUrl() == 'about:blank' this.setState 'awaiting_frame_load' else this.sendResponse(true) else if new Date().getTime() < timeout setTimeout((=> this.push_frame(name, timeout)), 50) else @owner.sendError(new Poltergeist.FrameNotFound(name)) pop_frame: -> this.sendResponse(@page.popFrame()) push_window: (name) -> sub_page = @page.getPage(name) if sub_page if sub_page.currentUrl() == 'about:blank' sub_page.onLoadFinished = => sub_page.onLoadFinished = null this.push_window(name) else @page_stack.push(@page) @page = sub_page @page_id += 1 this.sendResponse(true) else @page_name = name this.setState 'awaiting_sub_page' pop_window: -> prev_page = @page_stack.pop() @page = prev_page if prev_page this.sendResponse(true) mouse_event: (page_id, id, name) -> # Get the node before changing state, in case there is an exception node = this.node(page_id, id) # If the event triggers onNavigationRequested, we will transition to the 'loading' # state and wait for onLoadFinished before sending a response. this.setState 'mouse_event' @last_mouse_event = node.mouseEvent(name) setTimeout => if @state != 'loading' this.setState 'default' this.sendResponse(@last_mouse_event) , 5 click: (page_id, id) -> this.mouse_event page_id, id, 'click' double_click: (page_id, id) -> this.mouse_event page_id, id, 'doubleclick' hover: (page_id, id) -> this.mouse_event page_id, id, 'mousemove' click_coordinates: (x, y) -> @page.sendEvent('click', x, y) this.sendResponse({ click: { x: x, y: y } }) drag: (page_id, id, other_id) -> this.node(page_id, id).dragTo this.node(page_id, other_id) this.sendResponse(true) trigger: (page_id, id, event) -> this.node(page_id, id).trigger(event) this.sendResponse(event) equals: (page_id, id, other_id) -> this.sendResponse this.node(page_id, id).isEqual(this.node(page_id, other_id)) reset: -> this.resetPage() this.sendResponse(true) render: (path, full) -> dimensions = @page.validatedDimensions() document = dimensions.document viewport = dimensions.viewport if full @page.setScrollPosition(left: 0, top: 0) @page.setClipRect(left: 0, top: 0, width: document.width, height: document.height) @page.render(path) @page.setScrollPosition(left: dimensions.left, top: dimensions.top) else @page.setClipRect(left: 0, top: 0, width: viewport.width, height: viewport.height) @page.render(path) this.sendResponse(true) resize: (width, height) -> @page.setViewportSize(width: width, height: height) this.sendResponse(true) network_traffic: -> this.sendResponse(@page.networkTraffic()) set_headers: (headers) -> # Workaround for https://code.google.com/p/phantomjs/issues/detail?id=745 @page.setUserAgent(headers['User-Agent']) if headers['User-Agent'] @page.setCustomHeaders(headers) this.sendResponse(true) response_headers: -> this.sendResponse(@page.responseHeaders()) cookies: -> this.sendResponse(@page.cookies()) # We're using phantom.addCookie so that cookies can be set # before the first page load has taken place. set_cookie: (cookie) -> phantom.addCookie(cookie) this.sendResponse(true) remove_cookie: (name) -> @page.deleteCookie(name) this.sendResponse(true) cookies_enabled: (flag) -> phantom.cookiesEnabled = flag this.sendResponse(true) set_js_errors: (value) -> @js_errors = value this.sendResponse(true) set_debug: (value) -> @_debug = value this.sendResponse(true) exit: -> phantom.exit() noop: -> # NOOOOOOP! # This command is purely for testing error handling browser_error: -> throw new Error('zomg')