require 'edn'
require 'json'
# required for ruby 1.8
require 'enumerator'
module Calabash
module Cucumber
module UIA
def uia(command,options={})
res = http({:method => :post, :path => 'uia'}, {:command => command}.merge(options))
res = JSON.parse(res)
if res['outcome'] != 'SUCCESS'
raise "uia action failed because: #{res['reason']}\n#{res['details']}"
end
res['results'].first
end
def uia_query(*queryparts)
#TODO escape '\n etc in query
uia_handle_command(:query, queryparts)
end
def uia_query_el(*queryparts)
#TODO escape '\n etc in query
uia_handle_command(:queryEl, queryparts)
end
def uia_query_windows(*queryparts)
#TODO escape '\n etc in query
uia_handle_command(:queryWindows, queryparts)
end
def uia_names(*queryparts)
#TODO escape '\n etc in query
uia_handle_command(:names, queryparts)
end
def uia_tap(*queryparts)
uia_handle_command(:tap, queryparts)
end
def uia_tap_mark(mark)
uia_handle_command(:tapMark, mark)
end
def uia_tap_offset(offset)
uia_handle_command(:tapOffset, offset)
end
def uia_double_tap(*queryparts)
uia_handle_command(:doubleTap, queryparts)
end
def uia_double_tap_mark(mark)
uia_double_tap(:view, {:marked => mark})
end
def uia_double_tap_offset(offset)
uia_handle_command(:doubleTapOffset, offset)
end
def uia_two_finger_tap(*queryparts)
uia_handle_command(:twoFingerTap, queryparts)
end
def uia_two_finger_tap_offset(offset)
uia_handle_command(:twoFingerTapOffset, offset)
end
def uia_flick_offset(from, to)
uia_handle_command(:flickOffset, from, to)
end
def uia_touch_hold(duration, *queryparts)
uia_handle_command(:touchHold, duration, queryparts)
end
def uia_touch_hold_offset(duration, offset)
uia_handle_command(:touchHoldOffset, duration, offset)
end
def uia_pan(from_q, to_q)
uia_handle_command(:pan, from_q, to_q)
end
def uia_pan_offset(from, to, options)
uia_handle_command(:panOffset, from, to, options)
end
def uia_swipe(*queryparts)
uia_handle_command(:swipe, queryparts)
end
def uia_swipe_offset(offset, options)
uia_handle_command(:swipeOffset, offset, options)
end
def uia_pinch(*queryparts)
uia_handle_command(:pinch, queryparts)
end
def uia_pinch_offset(in_or_out, offset, duration)
uia_handle_command(:pinchOffset, in_or_out, offset, duration)
end
def uia_scroll_to(*queryparts)
uia_handle_command(:scrollTo, queryparts)
end
def uia_element_exists?(*queryparts)
uia_handle_command(:elementExists, queryparts)
end
def uia_element_does_not_exist?(*queryparts)
uia_handle_command(:elementDoesNotExist, queryparts)
end
def uia_screenshot(name)
uia_handle_command(:screenshot, name)
end
def uia_type_string(string, opt_text_before='', escape=true)
if escape && string.index(/\\/)
indexes = string.enum_for(:scan,/\\/).map { Regexp.last_match.begin(0) }
indexes.reverse.each { |idx| string = string.insert(idx, '\\') }
end
res = uia_handle_command(:typeString, string, opt_text_before)
status = res['status']
if status.eql?('error')
value = res['value']
raise "could not type '#{string}' - '#{value}'"
end
status
end
def uia_enter
uia_handle_command(:enter)
end
def uia_set_location(options)
validate_hash_is_location!(options)
if options[:place]
place = options[:place]
search_results = Geocoder.search(place)
raise "Got no results for #{place}" if search_results.empty?
loc = search_results.first
loc_data = {'latitude'=>loc.latitude, 'longitude'=>loc.longitude}
elsif options.is_a?(Hash)
loc_data = options
end
uia_handle_command(:setLocation, loc_data)
end
def uia_send_app_to_background(secs)
uia_handle_command(:deactivate, secs)
end
def uia_call(args_arr, *opts)
uia_call_method(:queryEl, args_arr, *opts)
end
def uia_call_windows(args_arr, *opts)
uia_call_method(:queryElWindows, args_arr, *opts)
end
def uia_call_method(cmd, args_arr, *opts)
if opts.empty?
return uia_handle_command(cmd, args_arr)
end
js_cmd = uia_serialize_command(cmd, args_arr)
js_args = []
opts.each do |invocation|
js_args << case invocation
when Symbol
"#{invocation}()"
when Hash
m = invocation.keys.first
args = invocation[m]
if args.is_a?(Array)
serialized_args = (args.map &:to_json).join(',')
else
serialized_args = args.to_json
end
"#{m}(#{serialized_args})"
else
raise "Invalid invocation spec #{invocation}"
end
end
command = "#{js_cmd}.#{js_args.join('.')}"
if ENV['DEBUG'] == '1'
puts 'Sending UIA command'
puts command
end
uia_result(uia(command))
end
def uia_handle_command(cmd, *query_args)
command = uia_serialize_command(cmd, *query_args)
if ENV['DEBUG'] == '1'
puts 'Sending UIA command'
puts command
end
s = uia(command)
uia_result(s)
end
def uia_serialize_command(cmd, *query_args)
args = uia_serialize_arguments(query_args)
%Q[uia.#{cmd}(#{args.join(', ')})]
end
def uia_serialize_arguments(args)
args.map do |part|
uia_serialize_argument(part)
end
end
def uia_serialize_argument(part)
if part.is_a?(String)
"'#{escape_uia_string(part)}'"
else
"'#{escape_uia_string(part.to_edn)}'"
end
end
def escape_uia_string(string)
#TODO escape '\n in query
escape_quotes string
end
# DEPRECATED: Use uia("...javascript..", options) instead.
# deprecated because the method signature is poor
def send_uia_command(opts ={})
# TODO formally deprecate send_uia_command with _deprecated function
#cmd = opts[:command]
#new_opts = cmd.select{|x| x != :command}
#_deprecated('0.9.163',
# "use 'uia(#{cmd}, #{new_opts})' instead",
# :warn)
uia(opts[:command], opts)
end
private
def validate_hash_is_location!(options)
return if options[:latitude] and options[:longitude]
if (options[:latitude] and not options[:longitude]) ||
(options[:longitude] and not options[:latitude])
raise 'Both latitude and longitude must be specified if either is.'
elsif not options[:place]
raise 'Either :place or :latitude and :longitude must be specified.'
end
end
def uia_result(s)
if ENV['DEBUG'] == '1'
puts 'Result'
p s
end
if s['status'] == 'success'
s['value']
else
s
end
end
end
end
end