e5cf8ff7fb
This is the initial release of iFreeRDP the FreeRDP iOS client. Build instructions can be found in docs/README.ios.
329 lines
12 KiB
Objective-C
329 lines
12 KiB
Objective-C
/*
|
|
RDP Touch Pointer View
|
|
|
|
Copyright 2013 Thinstuff Technologies GmbH, Author: Martin Fleisz
|
|
|
|
This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
|
|
If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#import "TouchPointerView.h"
|
|
#import "Utils.h"
|
|
|
|
|
|
#define RESET_DEFAULT_POINTER_IMAGE_DELAY 0.15
|
|
|
|
#define POINTER_ACTION_CURSOR 0
|
|
#define POINTER_ACTION_CLOSE 3
|
|
#define POINTER_ACTION_RCLICK 2
|
|
#define POINTER_ACTION_LCLICK 4
|
|
#define POINTER_ACTION_MOVE 4
|
|
#define POINTER_ACTION_SCROLL 5
|
|
#define POINTER_ACTION_KEYBOARD 7
|
|
#define POINTER_ACTION_EXTKEYBOARD 8
|
|
#define POINTER_ACTION_RESET 6
|
|
|
|
@interface TouchPointerView (Private)
|
|
- (void)setCurrentPointerImage:(UIImage*)image;
|
|
- (void)displayPointerActionImage:(UIImage*)image;
|
|
- (BOOL)pointInsidePointer:(CGPoint)point;
|
|
- (BOOL)pointInsidePointerArea:(int)area point:(CGPoint)point;
|
|
- (CGPoint)getCursorPosition;
|
|
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event;
|
|
- (void)handleSingleTap:(UITapGestureRecognizer*)gesture;
|
|
- (void)handlerForGesture:(UIGestureRecognizer*)gesture sendClick:(BOOL)sendClick;
|
|
@end
|
|
|
|
@implementation TouchPointerView
|
|
|
|
@synthesize delegate = _delegate;
|
|
|
|
- (void)awakeFromNib
|
|
{
|
|
[super awakeFromNib];
|
|
|
|
// set content mode when rotating (keep aspect ratio)
|
|
[self setContentMode:UIViewContentModeTopLeft];
|
|
|
|
// load touchPointerImage
|
|
_default_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_default" ofType:@"png"]] retain];
|
|
_active_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_active" ofType:@"png"]] retain];
|
|
_lclick_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_lclick" ofType:@"png"]] retain];
|
|
_rclick_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_rclick" ofType:@"png"]] retain];
|
|
_scroll_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_scroll" ofType:@"png"]] retain];
|
|
_extkeyboard_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_ext_keyboard" ofType:@"png"]] retain];
|
|
_keyboard_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_keyboard" ofType:@"png"]] retain];
|
|
_reset_pointer_img = [[UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:@"touch_pointer_reset" ofType:@"png"]] retain];
|
|
_cur_pointer_img = _default_pointer_img;
|
|
_pointer_transformation = CGAffineTransformMake(1, 0, 0, 1, 0, 0);
|
|
|
|
// init flags
|
|
_pointer_moving = NO;
|
|
_pointer_scrolling = NO;
|
|
|
|
// create areas array
|
|
int i, j;
|
|
CGFloat area_width = [_cur_pointer_img size].width / 3.0f;
|
|
CGFloat area_height = [_cur_pointer_img size].height / 3.0f;
|
|
for(i = 0; i < 3; i++)
|
|
{
|
|
for(j = 0; j < 3; j++)
|
|
{
|
|
_pointer_areas[j + i * 3] = CGRectMake(j * area_width, i * area_height, area_width, area_height);
|
|
}
|
|
}
|
|
|
|
// init gesture recognizers
|
|
UITapGestureRecognizer* singleTapRecognizer = [[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(handleSingleTap:)] autorelease];
|
|
[singleTapRecognizer setNumberOfTouchesRequired:1];
|
|
[singleTapRecognizer setNumberOfTapsRequired:1];
|
|
|
|
UILongPressGestureRecognizer* dragDropRecognizer = [[[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handleDragDrop:)] autorelease];
|
|
dragDropRecognizer.minimumPressDuration = 0.4;
|
|
// dragDropRecognizer.allowableMovement = 1000.0;
|
|
|
|
UILongPressGestureRecognizer* pointerMoveScrollRecognizer = [[[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handlePointerMoveScroll:)] autorelease];
|
|
pointerMoveScrollRecognizer.minimumPressDuration = 0.15;
|
|
pointerMoveScrollRecognizer.allowableMovement = 1000.0;
|
|
[pointerMoveScrollRecognizer requireGestureRecognizerToFail:dragDropRecognizer];
|
|
|
|
[self addGestureRecognizer:singleTapRecognizer];
|
|
[self addGestureRecognizer:dragDropRecognizer];
|
|
[self addGestureRecognizer:pointerMoveScrollRecognizer];
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[super dealloc];
|
|
[_default_pointer_img autorelease];
|
|
[_active_pointer_img autorelease];
|
|
[_lclick_pointer_img autorelease];
|
|
[_rclick_pointer_img autorelease];
|
|
[_scroll_pointer_img autorelease];
|
|
[_extkeyboard_pointer_img autorelease];
|
|
[_keyboard_pointer_img autorelease];
|
|
[_reset_pointer_img autorelease];
|
|
}
|
|
|
|
#pragma mark - Public interface
|
|
|
|
// positions the pointer on screen if it got offscreen after an orentation change
|
|
-(void)ensurePointerIsVisible
|
|
{
|
|
CGRect bounds = [self bounds];
|
|
if(_pointer_transformation.tx > (bounds.size.width - _cur_pointer_img.size.width))
|
|
_pointer_transformation.tx = bounds.size.width - _cur_pointer_img.size.width;
|
|
if(_pointer_transformation.ty > (bounds.size.height - _cur_pointer_img.size.height))
|
|
_pointer_transformation.ty = bounds.size.height - _cur_pointer_img.size.height;
|
|
[self setNeedsDisplay];
|
|
}
|
|
|
|
// show/hides the touch pointer
|
|
-(void)setHidden:(BOOL)hidden
|
|
{
|
|
[super setHidden:hidden];
|
|
|
|
// if shown center pointer in view
|
|
if(!hidden)
|
|
{
|
|
_pointer_transformation = CGAffineTransformMakeTranslation(([self bounds].size.width - [_cur_pointer_img size].width) / 2,
|
|
([self bounds].size.height - [_cur_pointer_img size].height) / 2);
|
|
[self setNeedsDisplay];
|
|
}
|
|
}
|
|
|
|
-(UIEdgeInsets)getEdgeInsets
|
|
{
|
|
return UIEdgeInsetsMake(0, 0, [_cur_pointer_img size].width, [_cur_pointer_img size].height);
|
|
}
|
|
|
|
- (CGPoint)getPointerPosition
|
|
{
|
|
return CGPointMake(_pointer_transformation.tx, _pointer_transformation.ty);
|
|
}
|
|
|
|
- (int)getPointerWidth
|
|
{
|
|
return [_cur_pointer_img size].width;
|
|
}
|
|
|
|
- (int)getPointerHeight
|
|
{
|
|
return [_cur_pointer_img size].height;
|
|
}
|
|
|
|
|
|
@end
|
|
|
|
@implementation TouchPointerView (Private)
|
|
|
|
- (void)setCurrentPointerImage:(UIImage*)image
|
|
{
|
|
_cur_pointer_img = image;
|
|
[self setNeedsDisplay];
|
|
}
|
|
|
|
- (void)displayPointerActionImage:(UIImage*)image
|
|
{
|
|
[self setCurrentPointerImage:image];
|
|
[self performSelector:@selector(setCurrentPointerImage:) withObject:_default_pointer_img afterDelay:RESET_DEFAULT_POINTER_IMAGE_DELAY];
|
|
}
|
|
|
|
// Only override drawRect: if you perform custom drawing.
|
|
// An empty implementation adversely affects performance during animation.
|
|
- (void)drawRect:(CGRect)rect
|
|
{
|
|
// Drawing code
|
|
CGContextRef context = UIGraphicsGetCurrentContext();
|
|
CGContextSaveGState(context);
|
|
CGContextConcatCTM(context, _pointer_transformation);
|
|
CGContextDrawImage(context, CGRectMake(0, 0, [_cur_pointer_img size].width, [_cur_pointer_img size].height), [_cur_pointer_img CGImage]);
|
|
CGContextRestoreGState(context);
|
|
}
|
|
|
|
// helper that returns YES if the given point is within the pointer
|
|
-(BOOL) pointInsidePointer:(CGPoint)point
|
|
{
|
|
CGRect rec = CGRectMake(0, 0, [_cur_pointer_img size].width, [_cur_pointer_img size].height);
|
|
return CGRectContainsPoint(CGRectApplyAffineTransform(rec, _pointer_transformation), point);
|
|
}
|
|
|
|
// helper that returns YES if the given point is within the given pointer area
|
|
-(BOOL) pointInsidePointerArea:(int)area point:(CGPoint)point
|
|
{
|
|
CGRect rec = _pointer_areas[area];
|
|
return CGRectContainsPoint(CGRectApplyAffineTransform(rec, _pointer_transformation), point);
|
|
}
|
|
|
|
// returns the position of the cursor
|
|
-(CGPoint) getCursorPosition
|
|
{
|
|
CGRect transPointerArea = CGRectApplyAffineTransform(_pointer_areas[POINTER_ACTION_CURSOR], _pointer_transformation);
|
|
return CGPointMake(CGRectGetMidX(transPointerArea), CGRectGetMidY(transPointerArea));
|
|
}
|
|
|
|
// this filters events - if the pointer was clicked the scrollview won't get any events
|
|
-(BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
|
|
{
|
|
return [self pointInsidePointer:point];
|
|
}
|
|
|
|
#pragma mark - Action handlers
|
|
|
|
// handles single tap gestures, returns YES if the event was handled by the pointer, NO otherwise
|
|
-(void) handleSingleTap:(UITapGestureRecognizer*)gesture
|
|
{
|
|
// get touch position within our view
|
|
CGPoint touchPos = [gesture locationInView:self];
|
|
|
|
// look if pointer was in one of our action areas
|
|
if([self pointInsidePointerArea:POINTER_ACTION_CLOSE point:touchPos])
|
|
[[self delegate] touchPointerClose];
|
|
else if([self pointInsidePointerArea:POINTER_ACTION_LCLICK point:touchPos])
|
|
{
|
|
[self displayPointerActionImage:_lclick_pointer_img];
|
|
[[self delegate] touchPointerLeftClick:[self getCursorPosition] down:YES];
|
|
[[self delegate] touchPointerLeftClick:[self getCursorPosition] down:NO];
|
|
}
|
|
else if([self pointInsidePointerArea:POINTER_ACTION_RCLICK point:touchPos])
|
|
{
|
|
[self displayPointerActionImage:_rclick_pointer_img];
|
|
[[self delegate] touchPointerRightClick:[self getCursorPosition] down:YES];
|
|
[[self delegate] touchPointerRightClick:[self getCursorPosition] down:NO];
|
|
}
|
|
else if([self pointInsidePointerArea:POINTER_ACTION_KEYBOARD point:touchPos])
|
|
{
|
|
[self displayPointerActionImage:_keyboard_pointer_img];
|
|
[[self delegate] touchPointerToggleKeyboard];
|
|
}
|
|
else if([self pointInsidePointerArea:POINTER_ACTION_EXTKEYBOARD point:touchPos])
|
|
{
|
|
[self displayPointerActionImage:_extkeyboard_pointer_img];
|
|
[[self delegate] touchPointerToggleExtendedKeyboard];
|
|
}
|
|
else if([self pointInsidePointerArea:POINTER_ACTION_RESET point:touchPos])
|
|
{
|
|
[self displayPointerActionImage:_reset_pointer_img];
|
|
[[self delegate] touchPointerResetSessionView];
|
|
}
|
|
}
|
|
|
|
-(void) handlerForGesture:(UIGestureRecognizer*)gesture sendClick:(BOOL)sendClick
|
|
{
|
|
if([gesture state] == UIGestureRecognizerStateBegan)
|
|
{
|
|
CGPoint touchPos = [gesture locationInView:self];
|
|
if([self pointInsidePointerArea:POINTER_ACTION_LCLICK point:touchPos])
|
|
{
|
|
_prev_touch_location = touchPos;
|
|
_pointer_moving = YES;
|
|
if(sendClick == YES)
|
|
{
|
|
[[self delegate] touchPointerLeftClick:[self getCursorPosition] down:YES];
|
|
[self setCurrentPointerImage:_active_pointer_img];
|
|
}
|
|
}
|
|
else if([self pointInsidePointerArea:POINTER_ACTION_SCROLL point:touchPos])
|
|
{
|
|
[self setCurrentPointerImage:_scroll_pointer_img];
|
|
_prev_touch_location = touchPos;
|
|
_pointer_scrolling = YES;
|
|
}
|
|
}
|
|
else if([gesture state] == UIGestureRecognizerStateChanged)
|
|
{
|
|
if(_pointer_moving)
|
|
{
|
|
CGPoint touchPos = [gesture locationInView:self];
|
|
_pointer_transformation = CGAffineTransformTranslate(_pointer_transformation, touchPos.x - _prev_touch_location.x, touchPos.y - _prev_touch_location.y);
|
|
[[self delegate] touchPointerMove:[self getCursorPosition]];
|
|
_prev_touch_location = touchPos;
|
|
[self setNeedsDisplay];
|
|
}
|
|
else if(_pointer_scrolling)
|
|
{
|
|
CGPoint touchPos = [gesture locationInView:self];
|
|
float delta = touchPos.y - _prev_touch_location.y;
|
|
if(delta > GetScrollGestureDelta())
|
|
{
|
|
[[self delegate] touchPointerScrollDown:YES];
|
|
_prev_touch_location = touchPos;
|
|
}
|
|
else if(delta < -GetScrollGestureDelta())
|
|
{
|
|
[[self delegate] touchPointerScrollDown:NO];
|
|
_prev_touch_location = touchPos;
|
|
}
|
|
}
|
|
}
|
|
else if([gesture state] == UIGestureRecognizerStateEnded)
|
|
{
|
|
if(_pointer_moving)
|
|
{
|
|
if(sendClick == YES)
|
|
[[self delegate] touchPointerLeftClick:[self getCursorPosition] down:NO];
|
|
_pointer_moving = NO;
|
|
[self setCurrentPointerImage:_default_pointer_img];
|
|
}
|
|
|
|
if(_pointer_scrolling)
|
|
{
|
|
[self setCurrentPointerImage:_default_pointer_img];
|
|
_pointer_scrolling = NO;
|
|
}
|
|
}
|
|
}
|
|
|
|
// handles long press gestures
|
|
-(void) handleDragDrop:(UILongPressGestureRecognizer*)gesture
|
|
{
|
|
[self handlerForGesture:gesture sendClick:YES];
|
|
}
|
|
|
|
-(void) handlePointerMoveScroll:(UILongPressGestureRecognizer*)gesture
|
|
{
|
|
[self handlerForGesture:gesture sendClick:NO];
|
|
}
|
|
|
|
@end
|