Как добавить лупу в пользовательский элемент управления?

Как добавить лупу в пользовательский элемент управления? Контроль - это дитя UIView. (Это UIWebView, но на некоторых страницах функция увеличения не работает.)

ОБНОВЛЕНО:

Может быть, можно заставить лупу использовать UIWebView?

2 ответа

Решение

1. Добавьте следующие файлы в ваш проект:

MagnifierView.h:

//
//  MagnifierView.h
//  SimplerMaskTest
//

#import <UIKit/UIKit.h>

@interface MagnifierView : UIView {
    UIView *viewToMagnify;
    CGPoint touchPoint;
}

@property (nonatomic, retain) UIView *viewToMagnify;
@property (assign) CGPoint touchPoint;

@end

MagnifierView.m:

//
//  MagnifierView.m
//  SimplerMaskTest
//

#import "MagnifierView.h"
#import <QuartzCore/QuartzCore.h>

@implementation MagnifierView
@synthesize viewToMagnify;
@dynamic touchPoint;

- (id)initWithFrame:(CGRect)frame {
    return [self initWithFrame:frame radius:118];
}

- (id)initWithFrame:(CGRect)frame radius:(int)r {
    int radius = r;

    if ((self = [super initWithFrame:CGRectMake(0, 0, radius, radius)])) {
        //Make the layer circular.
        self.layer.cornerRadius = radius / 2;
        self.layer.masksToBounds = YES;
    }

    return self;
}

- (void)setTouchPoint:(CGPoint)pt {
    touchPoint = pt;
    // whenever touchPoint is set, update the position of the magnifier (to just above what's being magnified)
    self.center = CGPointMake(pt.x, pt.y-66);
}

- (CGPoint)getTouchPoint {
    return touchPoint;
}

- (void)drawRect:(CGRect)rect {
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGRect bounds = self.bounds;
    CGImageRef mask = [UIImage imageNamed: @"loupe-mask@2x.png"].CGImage;
    UIImage *glass = [UIImage imageNamed: @"loupe-hi@2x.png"];

    CGContextSaveGState(context);
    CGContextClipToMask(context, bounds, mask);
    CGContextFillRect(context, bounds);
    CGContextScaleCTM(context, 1.2, 1.2);

    //draw your subject view here
    CGContextTranslateCTM(context,1*(self.frame.size.width*0.5),1*(self.frame.size.height*0.5));
    //CGContextScaleCTM(context, 1.5, 1.5);
    CGContextTranslateCTM(context,-1*(touchPoint.x),-1*(touchPoint.y));
    [self.viewToMagnify.layer renderInContext:context];

    CGContextRestoreGState(context);
    [glass drawInRect: bounds];
}

- (void)dealloc {
    [viewToMagnify release];
    [super dealloc];
}


@end

TouchReader.h:

//
//  TouchReader.h
//  SimplerMaskTest
//

#import <UIKit/UIKit.h>
#import "MagnifierView.h"

@interface TouchReader : UIView {
    NSTimer *touchTimer;
    MagnifierView *loop;
}

@property (nonatomic, retain) NSTimer *touchTimer;

- (void)addLoop;
- (void)handleAction:(id)timerObj;

@end

TouchReader.m:

//
//  TouchReader.m
//  SimplerMaskTest
//

#import "TouchReader.h"
#import "MagnifierView.h"

@implementation TouchReader

@synthesize touchTimer;

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
    self.touchTimer = [NSTimer scheduledTimerWithTimeInterval:0.5 target:self selector:@selector(addLoop) userInfo:nil repeats:NO];

    // just create one loop and re-use it.
    if (loop == nil) {
        loop = [[MagnifierView alloc] init];
        loop.viewToMagnify = self;
    }

    UITouch *touch = [touches anyObject];
    loop.touchPoint = [touch locationInView:self];
    [loop setNeedsDisplay];
}

- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event {
    [self handleAction:touches];
}

- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event {
    [self.touchTimer invalidate];
    self.touchTimer = nil;
    [loop removeFromSuperview];
}

- (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event {
    [self.touchTimer invalidate];
    self.touchTimer = nil;
    [loop removeFromSuperview];
}

- (void)addLoop {
    // add the loop to the superview.  if we add it to the view it magnifies, it'll magnify itself!
    [self.superview addSubview:loop];
    // here, we could do some nice animation instead of just adding the subview...
}

- (void)handleAction:(id)timerObj {
    NSSet *touches = timerObj;
    UITouch *touch = [touches anyObject];
    loop.touchPoint = [touch locationInView:self];
    [loop setNeedsDisplay];
}

- (void)dealloc {
    [loop release];
    loop = nil;
    [super dealloc];
}

@end

Основано на: http://coffeeshopped.com/2010/03/a-simpler-magnifying-glass-loupe-view-for-the-iphone

2. Добавьте следующие изображения:

Используемые изображения в коде:

loupe-hi@2x.png:

loupe-hi@2x.png

loupe-mask@2x.png:

loupe-mask@2x.png

Оригинальные, но центрированные изображения с тенью (в данный момент не используются):

loupe-shadow-hi@2x.png:

loupe-shadow-hi@2x.png

loupe-shadow-mask@2x.png:

loupe-shadow-mask@2x.png

3. Замените основной UIView в вашем xib-файле на TouchReader

Лупа будет работать автоматически, кроме элементов управления, которые сами фиксируют события касания (например, UIWebView). И код выше не поддерживает изображения с тенью. Пожалуйста, добавьте новый ответ в вопросе, если вы успешно исправили эту проблему.

ОБНОВЛЕНО:

Измените следующий код, чтобы добавить UIWebView служба поддержки. UIView должен остаться UIView,

@interface TouchReader : UILongPressGestureRecognizer

И добавить жест webView:

TouchReader* gestureMagnifier = [[[TouchReader alloc] initWithTarget:self action:@selector(handleMagnifier:)] autorelease];
gestureMagnifier.webView = editSource;
gestureMagnifier.delegate = self;
gestureMagnifier.minimumPressDuration = 0.5;
[webView addGestureRecognizer:gestureMagnifier];

TouchReader.h:

//- (void)handleAction:(id)timerObj;
-(void) handleGestureAction:(CGPoint)location;

TouchReader.m:

-(void)awakeFromNib
{
    UILongPressGestureRecognizer * longPressGesture = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handleGesture:)];
    [self addGestureRecognizer:longPressGesture];
}

-(void)handleGesture:(UILongPressGestureRecognizer *)longPressGesture
{

    CGPoint location = [longPressGesture locationInView:self];

    switch (longPressGesture.state) {
        case UIGestureRecognizerStateBegan:
            self.touchTimer = [NSTimer scheduledTimerWithTimeInterval:0.5
                                                               target:self
                                                             selector:@selector(addLoop)
                                                             userInfo:nil
                                                              repeats:NO];

            // just create one loop and re-use it.
            if(loop == nil){
                loop = [[MagnifierView alloc] init];
                loop.viewToMagnify = self;
            }

            loop.touchPoint = location;
            [loop setNeedsDisplay];
            break;

        case UIGestureRecognizerStateChanged:
            [self handleGestureAction:location];
            break;

        case UIGestureRecognizerStateEnded:
            [self.touchTimer invalidate];
            self.touchTimer = nil;

            [loop removeFromSuperview];
            loop=nil;
            break;

        default:
            break;
    }
}

- (void)addLoop {
    // add the loop to the superview.  if we add it to the view it magnifies, it'll magnify itself!
    [self.superview addSubview:loop];
}

-(void) handleGestureAction:(CGPoint)location
{
    loop.touchPoint = location;
    [loop setNeedsDisplay];

}

Вам не нужно touches... методов больше нет.

Другие вопросы по тегам