iOS--二維碼掃描與生成

一、掃描

1济似、? ZBar

ZBar在掃描的靈敏度上矫废,和內(nèi)存的使用上相對于ZXing上都是較優(yōu)的,但是對于 “圓角二維碼” 的掃描確很困難

2砰蠢、ZXing

ZXing 是 Google Code上的一個開源的條形碼掃描庫蓖扑,是用java設(shè)計的,連Google Glass 都在使用的台舱。但有人為了追求更高效率以及可移植性律杠,出現(xiàn)了c++ port. Github上的Objectivc-C port,其實就是用OC代碼封裝了一下而已竞惋,而且已經(jīng)停止維護(hù)柜去。這樣效率非常低,在instrument下面可以看到CPU和內(nèi)存瘋漲拆宛,在內(nèi)存小的機(jī)器上很容易崩潰嗓奢。

3、AVFoundation

AVFoundation無論在掃描靈敏度和性能上來說都是最優(yōu)的浑厚,所以毫無疑問我們應(yīng)該切換到AVFoundation股耽,需要兼容iOS 6或之前的版本可以用zbar或zxing代替。

代碼如下:

#import "PUCaptureCodeViewController.h"? ? #import@interface PUCaptureCodeViewController (){

AVCaptureSession *session;? //掃描會話

AVCaptureDeviceInput *deviceInput;? //輸入設(shè)備

AVCaptureMetadataOutput *metadataOutput; //輸出設(shè)備

AVCaptureVideoPreviewLayer *previewLayer;? //預(yù)覽圖層

}

@property (weak, nonatomic) IBOutlet UIImageView *edgeView;? //邊沿視圖

@property (weak, nonatomic) IBOutlet UITabBar *tabBar;

@property (weak, nonatomic) IBOutlet UIImageView *slinderView;? //沖擊波視圖

@property (weak, nonatomic) IBOutlet NSLayoutConstraint *sacnLineTop;? //沖擊波視圖與頂部的約束

@property (weak, nonatomic) IBOutlet NSLayoutConstraint *scanBGViewHight;

@end

@implementation PUCaptureCodeViewController

- (void)viewDidLoad {

[super viewDidLoad];

}

- (void)viewDidAppear:(BOOL)animated{

[super viewDidAppear:animated];

[self layoutUI];

[self startAnimation];

[self startScan];

}

pragma mark private Methods

- (void)layoutUI{

//設(shè)置默認(rèn)的 tabBar

self.tabBar.selectedItem = self.tabBar.items[0];

UIImage *image = [UIImage imageNamed:@"qrcode_border@2x"];

image = [image resizableImageWithCapInsets:UIEdgeInsetsMake(image.size.height/2, image.size.width/2,image.size.height/2, image.size.width/2) resizingMode:UIImageResizingModeStretch];

self.edgesForExtendedLayout = UIRectEdgeNone;

self.edgeView.image = image;

//1钳幅、獲取掃描會話

session = [[AVCaptureSession alloc]init];

//2. 獲取到輸入設(shè)備

AVCaptureDevice? *device? = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];

if (device != nil) {

deviceInput = [[AVCaptureDeviceInput alloc]initWithDevice:device error:nil];

}else{

return;

}

//3物蝙、獲取到輸出設(shè)備

metadataOutput = [[AVCaptureMetadataOutput alloc]init];

//4、獲取到預(yù)覽圖層

previewLayer = [[AVCaptureVideoPreviewLayer alloc]initWithSession:session];

previewLayer.frame = self.view.frame;

//5. 設(shè)置填充模式,不設(shè)置填充模式敢艰,4s可能會出現(xiàn)問題

previewLayer.videoGravity = AVLayerVideoGravityResizeAspectFill;

}

//開始掃描

- (void)startScan{

//1诬乞、判斷是否支持輸入設(shè)備

if(![session canAddInput:deviceInput]){

return;

}

//2、判斷是否支持輸出設(shè)備

if (![session canAddOutput:metadataOutput]) {

return;

}

//3、添加輸入設(shè)備

[session addInput:deviceInput];

//4. 添加輸出設(shè)備

[session addOutput:metadataOutput];

//5. 設(shè)置輸出設(shè)備支持的類型,availableMetadataObjectTypes 為設(shè)備支持的所有類型

metadataOutput.metadataObjectTypes = metadataOutput.availableMetadataObjectTypes;

//6丽惭、設(shè)置代理击奶,用于獲取掃描的內(nèi)容

[metadataOutput setMetadataObjectsDelegate:self queue:dispatch_get_main_queue()];

//7辈双、添加圖層

[self.view.layer insertSublayer:previewLayer atIndex:0];

//8责掏、開始掃描

[session startRunning];

}

- (void)startAnimation{

//設(shè)置起始位置

self.sacnLineTop.constant = -300;

//更新約束

[self.view layoutIfNeeded];

//設(shè)置動畫

[UIView animateWithDuration:3.0 animations:^{

//設(shè)置重復(fù)次數(shù)

[UIView setAnimationRepeatCount:MAXFLOAT];

self.sacnLineTop.constant = self.view.frame.size.height;

[self.view layoutIfNeeded];

}];

}

pragma mark delegate Methods


- (void)captureOutput:(AVCaptureOutput *)captureOutput didOutputMetadataObjects:(NSArray *)metadataObjects fromConnection:(AVCaptureConnection *)connection{

NSLog(@"%@", metadataObjects.lastObject);

// 1. 如果掃描完成,停止會話

[session stopRunning];

// 2. 刪除預(yù)覽圖層

[previewLayer removeFromSuperlayer];

NSLog(@"%@", metadataObjects);

// 3. 設(shè)置界面顯示掃描結(jié)果

if (metadataObjects.count > 0){

AVMetadataMachineReadableCodeObject *obj = metadataObjects[0];

// 提示:如果需要對url或者名片等信息進(jìn)行掃描湃望,可以在此進(jìn)行擴(kuò)展换衬!

NSLog(@"%@", obj.stringValue);

}

}

- (void)tabBar:(UITabBar *)tabBar didSelectItem:(UITabBarItem *)item{

self.tabBar.selectedItem = item;

self.scanBGViewHight.constant = item.tag == 0 ? 300 : 300*0.5;

[self.view layoutIfNeeded];

//移除掉所有動畫

[self.slinderView.layer removeAllAnimations];

//開始動畫

[self startAnimation];

}

@end

二、生成二維碼

1证芭、生成二維碼的步驟: 導(dǎo)入CoreImage框架 通過濾鏡CIFilter生成二維碼

二維碼的內(nèi)容(傳統(tǒng)的條形碼只能放數(shù)字):純文本,名片,URL

// 1. 實例化二維碼濾鏡

CIFilter *filter = [CIFilter filterWithName:@"CIQRCodeGenerator"];

// 2. 恢復(fù)濾鏡的默認(rèn)屬性

[filter setDefaults];

// 3. 將字符串轉(zhuǎn)換成

NSData NSData *data = [str dataUsingEncoding:NSUTF8StringEncoding];

// 4. 通過KVO設(shè)置濾鏡inputMessage數(shù)據(jù)

[filter setValue:data forKey:@"inputMessage"];

// 5. 獲得濾鏡輸出的圖像***

CIImage *outputImage = [filter outputImage];

// 6. 將CIImage轉(zhuǎn)換成UIImage瞳浦,并放大顯示

return [UIImage imageWithCIImage:outputImage scale:20.0 orientation:UIImageOrientationUp];

2、生成二維碼實例


- (void)viewDidLoad {

[super viewDidLoad];

// Do any additional setup after loading the view, typically from a nib.

self.imgView=[[UIImageView alloc]initWithFrame:CGRectMake(100, 100, [UIScreen mainScreen].bounds.size.width/2.0, [UIScreen mainScreen].bounds.size.width/2.0)];

[self.view addSubview:_imgView];

[self erweima];

}

pragma mark private Methods

-(void)erweima

{

//二維碼濾鏡

CIFilter *filter=[CIFilter filterWithName:@"CIQRCodeGenerator"];

//恢復(fù)濾鏡的默認(rèn)屬性

[filter setDefaults];

//將字符串轉(zhuǎn)換成NSData

NSData *data=[@"www.baidu.com" dataUsingEncoding:NSUTF8StringEncoding];

//通過KVO設(shè)置濾鏡inputmessage數(shù)據(jù)

[filter setValue:data forKey:@"inputMessage"];

//獲得濾鏡輸出的圖像

CIImage *outputImage=[filter outputImage];

//將CIImage轉(zhuǎn)換成UIImage,并放大顯示

_imgView.image=[self createNonInterpolatedUIImageFormCIImage:outputImage withSize:100.0];

//如果還想加上陰影废士,就在ImageView的Layer上使用下面代碼添加陰影

_imgView.layer.shadowOffset=CGSizeMake(0, 0.5);//設(shè)置陰影的偏移量

_imgView.layer.shadowRadius=1;//設(shè)置陰影的半徑

_imgView.layer.shadowColor=[UIColor blackColor].CGColor;//設(shè)置陰影的顏色為黑色

_imgView.layer.shadowOpacity=0.3;

}

//改變二維碼大小

- (UIImage *)createNonInterpolatedUIImageFormCIImage:(CIImage *)image withSize:(CGFloat) size {

CGRect extent = CGRectIntegral(image.extent);

CGFloat scale = MIN(size/CGRectGetWidth(extent), size/CGRectGetHeight(extent));

// 創(chuàng)建bitmap;

size_t width = CGRectGetWidth(extent) * scale;

size_t height = CGRectGetHeight(extent) * scale;

CGColorSpaceRef cs = CGColorSpaceCreateDeviceGray();

CGContextRef bitmapRef = CGBitmapContextCreate(nil, width, height, 8, 0, cs, (CGBitmapInfo)kCGImageAlphaNone);

CIContext *context = [CIContext contextWithOptions:nil];

CGImageRef bitmapImage = [context createCGImage:image fromRect:extent];

CGContextSetInterpolationQuality(bitmapRef, kCGInterpolationNone);

CGContextScaleCTM(bitmapRef, scale, scale);

CGContextDrawImage(bitmapRef, extent, bitmapImage);

// 保存bitmap到圖片

CGImageRef scaledImage = CGBitmapContextCreateImage(bitmapRef);

CGContextRelease(bitmapRef);

CGImageRelease(bitmapImage);

return [UIImage imageWithCGImage:scaledImage];

}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末叫潦,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子官硝,更是在濱河造成了極大的恐慌矗蕊,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,544評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件氢架,死亡現(xiàn)場離奇詭異傻咖,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)岖研,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評論 3 392
  • 文/潘曉璐 我一進(jìn)店門卿操,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人孙援,你說我怎么就攤上這事害淤。” “怎么了拓售?”我有些...
    開封第一講書人閱讀 162,764評論 0 353
  • 文/不壞的土叔 我叫張陵筝家,是天一觀的道長。 經(jīng)常有香客問我邻辉,道長溪王,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,193評論 1 292
  • 正文 為了忘掉前任值骇,我火速辦了婚禮莹菱,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘吱瘩。我一直安慰自己道伟,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,216評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著蜜徽,像睡著了一般祝懂。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上拘鞋,一...
    開封第一講書人閱讀 51,182評論 1 299
  • 那天砚蓬,我揣著相機(jī)與錄音,去河邊找鬼盆色。 笑死灰蛙,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的隔躲。 我是一名探鬼主播摩梧,決...
    沈念sama閱讀 40,063評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼宣旱!你這毒婦竟也來了仅父?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,917評論 0 274
  • 序言:老撾萬榮一對情侶失蹤浑吟,失蹤者是張志新(化名)和其女友劉穎笙纤,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體买置,經(jīng)...
    沈念sama閱讀 45,329評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡粪糙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,543評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了忿项。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蓉冈。...
    茶點故事閱讀 39,722評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖轩触,靈堂內(nèi)的尸體忽然破棺而出寞酿,到底是詐尸還是另有隱情,我是刑警寧澤脱柱,帶...
    沈念sama閱讀 35,425評論 5 343
  • 正文 年R本政府宣布伐弹,位于F島的核電站,受9級特大地震影響榨为,放射性物質(zhì)發(fā)生泄漏惨好。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,019評論 3 326
  • 文/蒙蒙 一随闺、第九天 我趴在偏房一處隱蔽的房頂上張望日川。 院中可真熱鬧,春花似錦矩乐、人聲如沸龄句。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,671評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽分歇。三九已至傀蓉,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間职抡,已是汗流浹背葬燎。 一陣腳步聲響...
    開封第一講書人閱讀 32,825評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留繁调,地道東北人萨蚕。 一個月前我還...
    沈念sama閱讀 47,729評論 2 368
  • 正文 我出身青樓靶草,卻偏偏與公主長得像蹄胰,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子奕翔,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,614評論 2 353

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