iOS开发-自定制图片浏览器

水墨丹青.jpg

前言:图片浏览器大家应该都用过,这方面的第三方也有很多,不过有时候第三方会跟我们的需求有一些出入,这就需要我们要么对第三方进行修改要么自己重新定制。我是比较喜欢自己重新定制的,在这给大家简单介绍一下我定制的图片浏览器,算是给大家提供一个思路,可以在此基础上进行修改完善。

原理:通过弹出UIViewController的形式来展示图片,使用UICollectionView并添加手势来实现图片浏览时图片的间隔。

首先创建一个继承于UIViewController的控制器,来作为图片浏览器的控制器,并实现相应的代码如下:

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

@interface RHPhotoBrowserController : UIViewController

- (instancetype)initWithType:(RHPhotoSourceType)type imageArr:(NSArray *)imageArr selectIndex:(NSInteger)selectIndex;
@end
#import "RHPhotoBrowserController.h"
#import "RHPhotoBrowserCell.h"

#define Cell_PhotoBrowser   @"Cell_PhotoBrowser"
#define PhotoSpace          10        // 图片间距
@interface RHPhotoBrowserController () <UICollectionViewDataSource, UICollectionViewDelegate, UICollectionViewDelegateFlowLayout>

@property (nonatomic, strong) UICollectionView * collection;
@property (nonatomic, strong) UIPageControl * pageControl;

@property (nonatomic, strong) NSMutableArray * dataArr;
@property (nonatomic, assign) RHPhotoSourceType type;
@property (nonatomic, assign) NSInteger selectIndex;

@property (nonatomic, assign) CGFloat panCenterX;
@property (nonatomic, assign) CGFloat startOffsetX;

@property (nonatomic, assign) CGFloat offsetX;
@property (nonatomic, assign) CGFloat panX;
@end

@implementation RHPhotoBrowserController

- (instancetype)initWithType:(RHPhotoSourceType)type imageArr:(NSArray *)imageArr selectIndex:(NSInteger)selectIndex {
    
    self = [super init];
    if (self) {
        
        [self.dataArr removeAllObjects];
        [self.dataArr addObjectsFromArray:imageArr];
        _type = type;
        _selectIndex = selectIndex;
    }
    return self;
}

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    [self addSubviews];
    [self makeConstraintsForUI];
}

#pragma mark - add subviews

- (void)addSubviews {
    
    self.view.backgroundColor = [UIColor blackColor];
    [self.view addSubview:self.collection];
    [self.view addSubview:self.pageControl];
}

- (void)makeConstraintsForUI {
    
    [_collection mas_makeConstraints:^(MASConstraintMaker *make) {
        
        make.top.left.right.bottom.mas_equalTo(0);
    }];
    [_pageControl mas_makeConstraints:^(MASConstraintMaker *make) {
        
        make.left.right.mas_equalTo(0);
        make.bottom.mas_equalTo(-SS(50));
        make.height.mas_equalTo(20);
    }];
    [self performSelector:@selector(setCollectionContentOffset) withObject:nil afterDelay:0.1];
}

- (void)setCollectionContentOffset {
    
    RHWeakSelf;
    dispatch_async(dispatch_get_main_queue(), ^{
        
        [weakSelf.collection setContentOffset:CGPointMake((Screen_Width + PhotoSpace) * _selectIndex, 0) animated:NO];
        weakSelf.pageControl.numberOfPages = weakSelf.dataArr.count;
        weakSelf.pageControl.currentPage = _selectIndex;
    });
    _startOffsetX = _collection.contentOffset.x;
}

#pragma mark - GestureRecognizer event

- (void)panCollection:(UIPanGestureRecognizer *)pan {
    
    _panCenterX = [pan translationInView:self.collection].x;
    if (pan.state == UIGestureRecognizerStateBegan) {
        
        _startOffsetX = _collection.contentOffset.x;
        _offsetX = 0;
        _panX = 0;
    }
    if (_selectIndex == 0) {
        
        if (_panCenterX > 0) {
            
            CGFloat s = (Screen_Width - _panCenterX) / Screen_Width;
            _offsetX += (_panCenterX - _panX) * s;
            _panX = _panCenterX;
            [self.collection setContentOffset:CGPointMake(-_offsetX, 0) animated:NO];
        } else {
            
            if (self.dataArr.count == 1) {
                
                CGFloat s = (Screen_Width + _panCenterX) / Screen_Width;
                _offsetX += (_panCenterX - _panX) * s;
                _panX = _panCenterX;
                [self.collection setContentOffset:CGPointMake(-_offsetX, 0) animated:NO];
            } else {
                
                [self.collection setContentOffset:CGPointMake(_startOffsetX - _panCenterX, 0) animated:NO];
            }
        }
    } else if (_selectIndex == self.dataArr.count - 1) {
        
        if (_panCenterX < 0) {
            
            CGFloat s = (Screen_Width + _panCenterX) / Screen_Width;
            _offsetX += (_panCenterX - _panX) * s;
            _panX = _panCenterX;
            [self.collection setContentOffset:CGPointMake(_startOffsetX - _offsetX, 0) animated:NO];
        } else {
            
            [self.collection setContentOffset:CGPointMake(_startOffsetX - _panCenterX, 0) animated:NO];
        }
    } else {
        
        [self.collection setContentOffset:CGPointMake(_startOffsetX - _panCenterX, 0) animated:NO];
    }
    if (pan.state == UIGestureRecognizerStateEnded) {
        
        if ([self absoluteValue:_panCenterX] > Screen_Width/3) {
            
            if (_panCenterX < 0) {
                
                _selectIndex += 1;
            } else {
                
                _selectIndex -= 1;
            }
            if (_selectIndex == self.dataArr.count) {
                
                _selectIndex = self.dataArr.count - 1;
            } else if (_selectIndex == -1) {
                
                _selectIndex = 0;
            }
            [self.collection setContentOffset:CGPointMake((Screen_Width + PhotoSpace) * _selectIndex, 0) animated:YES];
            self.pageControl.currentPage = _selectIndex;
        } else {
            
            [self.collection setContentOffset:CGPointMake(_startOffsetX, 0) animated:YES];
        }
    }
}

- (void)swipeCollection:(UISwipeGestureRecognizer *)swipe {
    
    if (swipe.direction == UISwipeGestureRecognizerDirectionLeft) {
        
        _selectIndex += 1;
    } else if (swipe.direction == UISwipeGestureRecognizerDirectionRight) {
        
        _selectIndex -= 1;
    }
    if (_selectIndex == self.dataArr.count) {
        
        _selectIndex = self.dataArr.count - 1;
    } else if (_selectIndex == -1) {
        
        _selectIndex = 0;
    }
    self.pageControl.currentPage = _selectIndex;
    [self.collection setContentOffset:CGPointMake((Screen_Width + PhotoSpace) * _selectIndex, 0) animated:YES];
}

// 返回value的绝对值
- (CGFloat)absoluteValue:(CGFloat)value {
    
    if (value < 0) {
        
        return -value;
    }
    return value;
}
#pragma mark - collection delegate

- (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section {
    
    return self.dataArr.count;
}

- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath {
    
    RHPhotoBrowserCell * cell = [collectionView dequeueReusableCellWithReuseIdentifier:Cell_PhotoBrowser forIndexPath:indexPath];
    if (indexPath.row < self.dataArr.count) {
        
        if (_type == RHPhotoSourceTypeImage) {
            
            UIImage * image = [self.dataArr objectAtIndex:indexPath.row];
            [cell configCellWithImage:image];
        } else if (_type == RHPhotoSourceTypeUrl) {
            
            NSString * url = [self.dataArr objectAtIndex:indexPath.row];
            [cell configCellWithUrl:url];
        } else if (_type == RHPhotoSourceTypeFilePath) {
            
            NSString * filePath = [self.dataArr objectAtIndex:indexPath.row];
            [cell configCellWithFilePath:filePath];
        } else if (_type == RHPhotoSourceTypeFileName) {
            
            NSString * fileName = [self.dataArr objectAtIndex:indexPath.row];
            [cell configCellWithFileName:fileName];
        }
    }
    return cell;
}

- (CGSize)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout *)collectionViewLayout sizeForItemAtIndexPath:(NSIndexPath *)indexPath {
    
    return CGSizeMake(Screen_Width, Screen_Height);
}

- (CGFloat)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout *)collectionViewLayout minimumLineSpacingForSectionAtIndex:(NSInteger)section {
    
    return PhotoSpace;
}

- (CGFloat)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout *)collectionViewLayout minimumInteritemSpacingForSectionAtIndex:(NSInteger)section {
    
    return 0;
}

- (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath {
    
    [self dismissViewControllerAnimated:YES completion:nil];
}

#pragma mark - setter and getter

- (UICollectionView *)collection {
    
    if (!_collection) {
        
        UICollectionViewFlowLayout * layout = [[UICollectionViewFlowLayout alloc] init];
        layout.scrollDirection = UICollectionViewScrollDirectionHorizontal;
        
        UICollectionView * cv = [[UICollectionView alloc] initWithFrame:CGRectZero collectionViewLayout:layout];
        cv.backgroundColor = [UIColor blackColor];
        cv.delegate = self;
        cv.dataSource = self;
        cv.showsHorizontalScrollIndicator = NO;
        [cv registerClass:[RHPhotoBrowserCell class] forCellWithReuseIdentifier:Cell_PhotoBrowser];
        UIPanGestureRecognizer * pan = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(panCollection:)];
        [cv addGestureRecognizer:pan];
        UISwipeGestureRecognizer * swipeL = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(swipeCollection:)];
        swipeL.direction = UISwipeGestureRecognizerDirectionLeft;
        [cv addGestureRecognizer:swipeL];
        UISwipeGestureRecognizer * swipeR = [[UISwipeGestureRecognizer alloc] initWithTarget:self action:@selector(swipeCollection:)];
        swipeR.direction = UISwipeGestureRecognizerDirectionRight;
        [cv addGestureRecognizer:swipeR];
        _collection = cv;
    }
    return _collection;
}

- (UIPageControl *)pageControl {
    
    if (!_pageControl) {
        
        UIPageControl * pageControl = [[UIPageControl alloc] init];
        pageControl.pageIndicatorTintColor = [[UIColor lightGrayColor] colorWithAlphaComponent:0.9];
        pageControl.currentPageIndicatorTintColor = [UIColor whiteColor];
        pageControl.userInteractionEnabled = NO;
        _pageControl = pageControl;
    }
    return _pageControl;
}

- (NSMutableArray *)dataArr {
    
    if (!_dataArr) {
        
        _dataArr = [NSMutableArray array];
    }
    return _dataArr;
}
@end

其实到此基本已经结束了,大家实现一个相对应的cell就可以了。使用时直接通过外漏的方法创建该控制器对象并弹出该控制器即可。

为了更加方便的调用,我又增加了一个NSObject的类来控制以上控制器的调用。如下:

#import <Foundation/Foundation.h>

typedef NS_ENUM(NSUInteger, RHPhotoSourceType) {
    
    RHPhotoSourceTypeImage     = 0,
    RHPhotoSourceTypeUrl       = 1,
    RHPhotoSourceTypeFilePath  = 2,
    RHPhotoSourceTypeFileName  = 3
};
@interface RHPhotoBrowser : NSObject

+ (RHPhotoBrowser *)shared;

- (void)browseImageWithType:(RHPhotoSourceType)type imageArr:(NSArray *)imageArr selectIndex:(NSInteger)selectIndex;
@end
#import "RHPhotoBrowser.h"
#import "RHPhotoBrowserController.h"

@implementation RHPhotoBrowser

+ (RHPhotoBrowser *)shared {
    
    static RHPhotoBrowser * helper = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        
        helper = [[RHPhotoBrowser alloc] init];
    });
    return helper;
}

- (void)browseImageWithType:(RHPhotoSourceType)type imageArr:(NSArray *)imageArr selectIndex:(NSInteger)selectIndex {
    
    if (selectIndex > imageArr.count - 1) {
        
        selectIndex = 0;
    }
    UIViewController * rootVC = [UIApplication sharedApplication].delegate.window.rootViewController;
    RHPhotoBrowserController * browser = [[RHPhotoBrowserController alloc] initWithType:type imageArr:imageArr selectIndex:selectIndex];
    [rootVC presentViewController:browser animated:YES completion:nil];
}
@end

这样使用的时候只需要使用该类就可以了。这里大家可以将单例去掉,将对象方法直接改为类方法即可。我是习惯了,所以这样写了。

再给大家看一下使用方法一步调用:

[[RHPhotoBrowser shared] browseImageWithType:RHPhotoSourceTypeFileName imageArr:@[@"c006", @"c007", @"c008", @"c009", @"c010"] selectIndex:2];

效果如下:


最后,还是希望能够帮助到有需要的朋友们,愿我们能够一起学习进步,在开发的道路上越走越顺利!!!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容