iOS實(shí)現(xiàn)頭像裁剪( 方或圓)功能疆偿,支持縮放拖拽咱筛。

類似微信頭像選擇裁剪功能 ,設(shè)置UIImagePickerController的allowsEditing屬性為YES即可實(shí)現(xiàn)翁脆,但如何自己實(shí)現(xiàn)此功能呢眷蚓?如何實(shí)現(xiàn)類似系統(tǒng)聯(lián)系人圓形頭像裁剪選取呢?

如圖:


Simulator Screen Shot 00.png
圖片要支持縮放和拖拽反番,首先想到scrollView沙热,具體實(shí)現(xiàn)方法如下:
//CropImageController.h
@protocol CropImageDelegate <NSObject>
- (void)cropImageDidFinishedWithImage:(UIImage *)image;
@end

@interface CropImageController : UIViewController

@property (nonatomic, weak) id <CropImageDelegate> delegate;
//圓形裁剪,默認(rèn)NO;
@property (nonatomic, assign) BOOL ovalClip;
- (instancetype)initWithImage:(UIImage *)originalImage delegate:(id)delegate;
@end

在viewDidLoad初始化scrollView,設(shè)置frame為裁剪框大小罢缸,若需要超出邊界線內(nèi)容仍然可以顯示篙贸,設(shè)置其layer屬性masksToBounds即可

- (void)viewDidLoad {
    [super viewDidLoad];
// Do any additional setup after loading the view.
    self.view.backgroundColor = [UIColor blackColor];
    self.automaticallyAdjustsScrollViewInsets = NO;
    CGFloat height = (ScreenHeight - ScreenWidth)/2.0;
    _scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, height ,ScreenWidth,ScreenWidth)];
    _scrollView.bouncesZoom = YES;
    _scrollView.minimumZoomScale = 1;
    _scrollView.maximumZoomScale = 3;
    _scrollView.zoomScale = 1;
    _scrollView.delegate = self;
    _scrollView.layer.masksToBounds = NO;
    _scrollView.showsHorizontalScrollIndicator = NO;
    _scrollView.showsVerticalScrollIndicator = NO;
    _scrollView.layer.borderWidth = 1.5;
    _scrollView.layer.borderColor = [UIColor whiteColor].CGColor;
      if (_ovalClip) {
        _scrollView.layer.cornerRadius = ScreenWidth/2.0;
    }
    self.view.layer.masksToBounds = YES;
    if (_originalImage) {
        _imageView = [[UIImageView alloc] initWithImage:_originalImage];
        CGFloat img_width = ScreenWidth;
        CGFloat img_height = _originalImage.size.height * (img_width/_originalImage.size.width);
        CGFloat img_y= (img_height - self.view.bounds.size.width)/2.0;
        _imageView.frame = CGRectMake(0,0, img_width, img_height);
        _imageView.userInteractionEnabled = YES;
        [_scrollView addSubview:_imageView];
        
        
        _scrollView.contentSize = CGSizeMake(img_width, img_height);
        _scrollView.contentOffset = CGPointMake(0, img_y);
        [self.view addSubview:_scrollView];
    }
    [self userInterface];
}

設(shè)置self.view中間鏤空效果:

- (void)userInterface {
    
    CGRect cropframe = _scrollView.frame;
    UIBezierPath * path = [UIBezierPath bezierPathWithRoundedRect:self.view.bounds cornerRadius:0];
    UIBezierPath * cropPath = [UIBezierPath bezierPathWithRoundedRect:cropframe cornerRadius:0];
    if (_ovalClip) {
        cropPath = [UIBezierPath bezierPathWithOvalInRect:cropframe];
    }
    [path appendPath:cropPath];
    
    CAShapeLayer * layer = [[CAShapeLayer alloc] init];
    layer.fillColor = [UIColor colorWithRed:.0 green:.0 blue:.0 alpha:0.5].CGColor;
    //填充規(guī)則
    layer.fillRule=kCAFillRuleEvenOdd;
    layer.path = path.CGPath;
    [self.view.layer addSublayer:layer];
}

UIScrollViewDelegate,縮放和滑動(dòng)過程中調(diào)整內(nèi)容視圖大小:

#pragma mark -- UIScrollViewDelegate
- (UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView {
    return self.imageView;
}
- (void)scrollViewDidZoom:(UIScrollView *)scrollView {
    //調(diào)整位置
    [self centerContent];
}
- (void)centerContent {
    CGRect imageViewFrame = _imageView.frame;
    
    CGRect scrollBounds = CGRectMake(0, 0, ScreenWidth, ScreenWidth);
    if (imageViewFrame.size.height > scrollBounds.size.height) {
        imageViewFrame.origin.y = 0.0f;
    }else {
        imageViewFrame.origin.y = (scrollBounds.size.height - imageViewFrame.size.height) / 2.0;
    }
    if (imageViewFrame.size.width < scrollBounds.size.width) {
        imageViewFrame.origin.x = (scrollBounds.size.width - imageViewFrame.size.width) /2.0;
    }else {
        imageViewFrame.origin.x = 0.0f;
    }
    _imageView.frame = imageViewFrame;
}

調(diào)整位置確定選取圖片時(shí)候枫疆,可根據(jù)當(dāng)前ScrollView的contentOffset和縮放倍數(shù)確定具體裁剪位置:

- (UIImage *)cropImage {
    CGPoint offset = _scrollView.contentOffset;
    //圖片縮放比例
    CGFloat zoom = _imageView.frame.size.width/_originalImage.size.width;
    //視網(wǎng)膜屏設(shè)備像素相關(guān)
    zoom = zoom / [UIScreen mainScreen].scale;
    
    CGFloat width = _scrollView.frame.size.width;
    CGFloat height = _scrollView.frame.size.height;
    if (_imageView.frame.size.height < _scrollView.frame.size.height) {//太胖了,取中間部分
        offset = CGPointMake(offset.x + (width - _imageView.frame.size.height)/2.0, 0);
        width = height = _imageView.frame.size.height;
    }
    
    CGRect rec = CGRectMake(offset.x/zoom, offset.y/zoom,width/zoom,height/zoom);
    CGImageRef imageRef =CGImageCreateWithImageInRect([_originalImage CGImage],rec);
    UIImage * image = [[UIImage alloc]initWithCGImage:imageRef];
    CGImageRelease(imageRef);
    if (_ovalClip) {
        image = [image ovalClip];
    }
    return image;
}

UIImage分類(圖片指定尺寸縮放和圓形裁剪):

//UIImage+Crop.m
- (UIImage *)resizeImageWithSize:(CGSize)newSize {
    CGFloat newWidth = newSize.width;
    CGFloat newHeight = newSize.height;
    float width  = self.size.width;
    float height = self.size.height;
    if (width != newWidth || height != newHeight) {
        UIGraphicsBeginImageContextWithOptions(CGSizeMake(newWidth, newHeight), YES, [UIScreen mainScreen].scale);
        [self drawInRect:CGRectMake(0, 0, newWidth, newHeight)];
        
        UIImage *resized = UIGraphicsGetImageFromCurrentImageContext();
        UIGraphicsEndImageContext();
        return resized;
    }
    return self;
}
- (UIImage *)ovalClip {

    CGSize size = self.size;
    UIGraphicsBeginImageContextWithOptions(size, NO, [UIScreen mainScreen].scale);
    UIBezierPath *path = [UIBezierPath bezierPathWithOvalInRect:CGRectMake(0, 0, self.size.width, self.size.height)];
    [path addClip];
    [self drawAtPoint:CGPointZero];
    UIImage * image = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    
    return image;
}

注:由于對(duì)圖片進(jìn)行裁剪爵川,先將原始圖片等比縮放到指定裁剪寬度。
源碼下載
CropImage

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末息楔,一起剝皮案震驚了整個(gè)濱河市寝贡,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌值依,老刑警劉巖圃泡,帶你破解...
    沈念sama閱讀 206,602評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異愿险,居然都是意外死亡颇蜡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,442評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門辆亏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來风秤,“玉大人,你說我怎么就攤上這事扮叨$拖遥” “怎么了?”我有些...
    開封第一講書人閱讀 152,878評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵彻磁,是天一觀的道長(zhǎng)碍沐。 經(jīng)常有香客問我惦费,道長(zhǎng),這世上最難降的妖魔是什么抢韭? 我笑而不...
    開封第一講書人閱讀 55,306評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮恍箭,結(jié)果婚禮上刻恭,老公的妹妹穿的比我還像新娘。我一直安慰自己扯夭,他們只是感情好鳍贾,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,330評(píng)論 5 373
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著交洗,像睡著了一般骑科。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上构拳,一...
    開封第一講書人閱讀 49,071評(píng)論 1 285
  • 那天咆爽,我揣著相機(jī)與錄音,去河邊找鬼置森。 笑死斗埂,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的凫海。 我是一名探鬼主播呛凶,決...
    沈念sama閱讀 38,382評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼行贪!你這毒婦竟也來了漾稀?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,006評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤建瘫,失蹤者是張志新(化名)和其女友劉穎崭捍,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體暖混,經(jīng)...
    沈念sama閱讀 43,512評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡缕贡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,965評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了拣播。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片晾咪。...
    茶點(diǎn)故事閱讀 38,094評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖贮配,靈堂內(nèi)的尸體忽然破棺而出谍倦,到底是詐尸還是另有隱情,我是刑警寧澤泪勒,帶...
    沈念sama閱讀 33,732評(píng)論 4 323
  • 正文 年R本政府宣布昼蛀,位于F島的核電站宴猾,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏叼旋。R本人自食惡果不足惜仇哆,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,283評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望夫植。 院中可真熱鬧讹剔,春花似錦、人聲如沸详民。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,286評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)沈跨。三九已至由捎,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間饿凛,已是汗流浹背狞玛。 一陣腳步聲響...
    開封第一講書人閱讀 31,512評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留笤喳,地道東北人为居。 一個(gè)月前我還...
    沈念sama閱讀 45,536評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像杀狡,于是被迫代替她去往敵國(guó)和親蒙畴。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,828評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫(kù)呜象、插件膳凝、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,029評(píng)論 4 62
  • 我們已經(jīng)知道,Gradle是一個(gè)非常靈活的構(gòu)建框架恭陡,他提供了構(gòu)建的基礎(chǔ)核心蹬音,為了對(duì)具體的業(yè)務(wù)進(jìn)行構(gòu)建,Gradle...
    acc8226閱讀 1,356評(píng)論 0 4
  • window.open("https://www.baidu.com")//打開新的窗口 說是有4個(gè)參數(shù)windo...
    貝灬小暉閱讀 817評(píng)論 0 0
  • Y0402Y閱讀 195評(píng)論 0 0
  • 上回說到天子率領(lǐng)祭祀的隊(duì)伍出城休玩,或者著淆,天子被打著祭祀名義的隊(duì)伍挾持出城了。 挾持拴疤,噢不永部,護(hù)衛(wèi)天子的隊(duì)伍是越騎營(yíng)。原...
    搬磚師爺閱讀 583評(píng)論 22 1