require 'spec_helper' require 'win/gui/input' require 'win/error' include WinTestApp include Win::Gui::Message include Win::Gui::Window include Win::Gui::Input include Win::Error def buffer @buffer ||= FFI::MemoryPointer.new :char, 1024 end def msg @msg ||= Win::Gui::Message::Msg.new end def msg_callback lambda { |handle, message, data, result| @handle = handle; @message = message; @data = data; @result = result } end def should_have msg, members members.each do |member, value| case member when :l_param msg[member].address.should == value when :time msg[member].should be > value else msg[member].should == value end end end def clear_thread_queue get_message while peek_message end describe Win::Gui::Message, ' defines a set of API functions related to Window messaging' do before(:all) { clear_thread_queue } describe '#post_message' do before(:each) { clear_thread_queue } after(:all) { close_test_app if @launched_test_app } spec { use { success = PostMessage(handle = 0, msg = 0, w_param = 0, l_param = nil) } } spec { use { success = post_message(handle = 0, msg = 0, w_param = 0, l_param = nil) } } context 'places (posts) a message in the message queue associated with the thread that created the specified window' do it 'with nil as last argument(lParam)' do app = launch_test_app post_message(app.handle, WM_SYSCOMMAND, SC_CLOSE, nil).should == true sleep SLEEP_DELAY window?(app.handle).should == false end it 'with pointer as last argument(lParam)' do app = launch_test_app post_message(app.handle, WM_SYSCOMMAND, SC_CLOSE, FFI::MemoryPointer.new(:long)).should == true sleep SLEEP_DELAY window?(app.handle).should == false end it 'with Fixnum as last argument(lParam)' do app = launch_test_app post_message(app.handle, WM_SYSCOMMAND, SC_CLOSE, 0).should == true sleep SLEEP_DELAY window?(app.handle).should == false end end it 'places (posts) a message into current thread`s queue if first arg is 0' do post_message(0, WM_USER, 33, nil).should == true msg = get_message() should_have msg, hwnd: 0, message: WM_USER, w_param: 33, l_param: 0 end it 'returns without waiting for the thread to process the message' end # describe '#post_message' describe '#send_message' do after(:all) { close_test_app if @launched_test_app } spec { use { success = SendMessage(handle = 0, msg = 0, w_param = 0, l_param = nil) } } spec { use { success = send_message(handle = 0, msg = 0, w_param = 0, l_param = nil) } } context 'sends a message to the specified window' do it 'with nil as last argument(lParam)' do app = launch_test_app send_message(app.handle, WM_SYSCOMMAND, SC_CLOSE, nil) sleep SLEEP_DELAY window?(app.handle).should == false end it 'with pointer as last argument(lParam)' do app = launch_test_app send_message(app.handle, WM_SYSCOMMAND, SC_CLOSE, FFI::MemoryPointer.new(:long)) sleep SLEEP_DELAY window?(app.handle).should == false end it 'with Fixnum as last argument(lParam)' do app = launch_test_app send_message(app.handle, WM_SYSCOMMAND, SC_CLOSE, 0) sleep SLEEP_DELAY window?(app.handle).should == false end end end # describe '#send_message' describe "#send_message_callback" do before(:all) { @app=launch_test_app } after(:all) { close_test_app if @launched_test_app } spec { use { success = SendMessageCallback(h_wnd=0, msg=0, w_param=0, l_param=nil, msg_callback, data=0) } } spec { use { success = send_message_callback(h_wnd=0, msg=0, w_param=0, l_param=nil, data=0, &msg_callback) } } it "sends message to window and returns, specifying callback to be called by system after message is processed" do sent = SendMessageCallback(@app.handle, WM_USER, 0, nil, msg_callback, data=13) sent.should == 1 @handle.should == nil # Callback did not fire just yet @message.should == nil @data.should == nil @result.should == nil sleep SLEEP_DELAY # Small delay to allow message delivery peek_message # Dispatching sent message (even though there is nothing in queue) @handle.should == @app.handle @message.should == WM_USER @data.should == 13 @result.should == 0 end it "snake_case api defaults data to 0, converts block into callback and returns true/false" do sent = send_message_callback(@app.handle, WM_USER, 0, nil) { |*args| @data=args[2] } sent.should == true @data.should == nil # Callback did not fire just yet sleep SLEEP_DELAY # Small delay to allow message delivery peek_message # Dispatching sent message (even though there is nothing in queue) @data.should == 0 end it "can be used with :pointer as a 4th arg" do sent = send_message_callback(@app.handle, WM_USER, 0, FFI::MemoryPointer.new(:long)) { |*args|} sent.should == true end it "can be used with :long as a 4th arg" do sent = send_message_callback(@app.handle, WM_USER, 0, 0) { |*args|} sent.should == true end it "fails if unable to send message" do sent = SendMessageCallback(not_a_handle, WM_USER, 0, nil, msg_callback, 0) sent.should == 0 send_message_callback(not_a_handle, WM_USER, 0, nil) { |*args| @data=args[2] } sent.should == 0 get_last_error.should == "Invalid window handle." end end # describe send_message_callback describe "#get_message" do before(:all) { clear_thread_queue; 2.times { post_message 0, 0, 0, nil } } spec { use { res = GetMessage(msg, handle=0, msg_filter_min=0, msg_filter_max=0) } } spec { use { message = get_message(msg, handle=0, msg_filter_min=0, msg_filter_max=0) } } it "original api retrieves a message from the calling thread's message queue" do set_cursor_pos(x=0, y=0) post_message(0, WM_USER+1, 33, nil) res = GetMessage(msg, handle=0, msg_filter_min=0, msg_filter_max=0) res.should == 1 # p msg[:hwnd], msg[:message], msg[:w_param], msg[:l_param], msg[:time], msg[:x], msg[:y] should_have msg, hwnd: 0, message: WM_USER+1, w_param: 33, x: 0, y: 0, l_param: 0, time: 1000000 end it "original api returns -1 if there is an error (wrong handle, in this case)" do res = GetMessage(msg, not_a_handle, msg_filter_min=0, msg_filter_max=0) res.should == -1 end it "original api returns 0 if WM_QUIT was posted to thread`s message queue" do post_message(0, WM_QUIT, 13, nil) res = GetMessage(msg, 0, msg_filter_min=0, msg_filter_max=0) res.should == 0 end it "snake_case api returns a message struct retrieved from the calling thread's message queue " do set_cursor_pos(x=99, y=99) post_message(0, WM_USER+2, 33, nil) msg = get_message() should_have msg, hwnd: 0, message: WM_USER+2, w_param: 33, x: 99, y: 99, l_param: 0, time: 1000000 end it "snake_case api returns nil if there is an error (wrong handle, in this case)" do get_message(msg, not_a_handle).should == nil end it "snake_case api returns false if WM_QUIT was posted to thread`s message queue" do post_message(0, WM_QUIT, 13, nil) get_message.should == false end end # describe get_message describe "#peek_message" do before(:all) { set_cursor_pos(x=0, y=0); post_message(0, WM_USER+2, 13, nil) } spec { use { success = PeekMessage(msg, h_wnd=0, filter_min=0, filter_max=0, remove_msg=0) } } spec { use { success = peek_message(msg, h_wnd=0, filter_min=0, filter_max=0, remove_msg=0) } } it "original api checks the thread message queue for a posted message, retrieves it without removing" do 10.times do res = PeekMessage(msg, h_wnd=0, filter_min=0, filter_max=0, remove_msg=0) res.should == 1 should_have msg, hwnd: 0, message: WM_USER+2, w_param: 13, x: 0, y: 0, l_param: 0, time: 1000000 end end it "snake_case api checks the thread message queue for a posted message, returns it without removing" do 10.times do msg = peek_message() should_have msg, hwnd: 0, message: WM_USER+2, w_param: 13, x: 0, y: 0, l_param: 0, time: 1000000 end end it "original api returns 0 if no message in queue" do get_message PeekMessage(msg, h_wnd=0, filter_min=0, filter_max=0, remove_msg=0).should == 0 end it "snake_case api returns nil if no message in queue" do peek_message.should == nil end end # describe peek_message describe "#translate_message" do spec { use { success = TranslateMessage(msg) } } spec { use { success = translate_message(msg) } } it "translates virtual-key message into character message which is then posted to the thread's message queue" do pending end it "returns zero/false if no translation took place" do TranslateMessage(msg).should == 0 translate_message(msg).should == false end end # describe translate_message describe "#dispatch_message" do spec { use { res = DispatchMessage(msg) } } #return value is normally ignored spec { use { res = dispatch_message(msg) } } #return value is normally ignored it "dispatches a message to a window procedure. Typically used to dispatch a message retrieved by GetMessage" do pending res = DispatchMessage(msg) end end # describe dispatch_message end # Win::Gui::Message, ' defines a set of API functions related to Window messaging'