使用UIImagePickerController 进行录制

#import "ViewController.h"
#import <MobileCoreServices/MobileCoreServices.h>
#import <QuartzCore/QuartzCore.h> @interface ViewController ()
<UIImagePickerControllerDelegate,UINavigationControllerDelegate> - (IBAction)videoRecod:(id)sender; @end @implementation ViewController - (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
} - (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
} - (IBAction)videoRecod:(id)sender { if ([UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypeCamera]) { UIImagePickerController *imagePickerController = [[UIImagePickerController alloc] init];
imagePickerController.delegate = self;
imagePickerController.sourceType = UIImagePickerControllerSourceTypeCamera; imagePickerController.mediaTypes = [[NSArray alloc]
initWithObjects:(NSString *)kUTTypeMovie, nil]; //录制质量设定
imagePickerController.videoQuality = UIImagePickerControllerQualityTypeHigh; //仅仅同意最多录制30秒时间
imagePickerController.videoMaximumDuration = 30.0f; [self presentViewController:imagePickerController animated:YES completion:nil]; } else {
NSLog(@"摄像头不可用。 ");
}
} - (void) imagePickerControllerDidCancel: (UIImagePickerController *) picker {
[self dismissViewControllerAnimated:YES completion:nil];
} - (void) imagePickerController: (UIImagePickerController *) picker
didFinishPickingMediaWithInfo: (NSDictionary *) info { NSURL *url = [info objectForKey:UIImagePickerControllerMediaURL];
NSString *tempFilePath = [url path]; if ( UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(tempFilePath) ) {
//video:didFinishSavingWithError:contextInfo: 必须保存成这类回调
UISaveVideoAtPathToSavedPhotosAlbum( tempFilePath,
self,
@selector(video:didFinishSavingWithError:contextInfo:),
(__bridge void *)(tempFilePath));
} [self dismissViewControllerAnimated:YES completion:nil]; } - (void)video:(NSString *)videoPath didFinishSavingWithError:(NSError *)error contextInfo:(NSString *)contextInfo {
NSString *title; NSString *message;
if (!error) {
title = @"视频保存";
message = @"视频已经保存到设备的相机胶卷中";
} else {
title = @"视频失败";
message = [error description];
}
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:title
message:message
delegate:nil
cancelButtonTitle:@"OK"
otherButtonTitles:nil];
[alert show];
} - (void)navigationController:(UINavigationController *)navigationController
willShowViewController:(UIViewController *)viewController
animated:(BOOL)animated
{
NSLog(@"选择器将要显示。");
} - (void)navigationController:(UINavigationController *)navigationController
didShowViewController:(UIViewController *)viewController
animated:(BOOL)animated
{
NSLog(@"选择器显示结束。 ");
}

使用AVFoundation

AVCaptureSession,捕获会话,是为了实现从摄像头和麦克风捕获数据,须要使用AVCaptureSession对象协调输入输出数据

AVCaptureDevice,捕获设备,代表输入一个设备,比如摄像头和麦克风

AVCaptureDeviceInput,捕获会话的一个输入数据源

AVCaptureOutput,捕获会话的一个输出目标,比如输出的视频文件和静态图片

AVCaptureMovieFileOutput,是AVCaptureOutput的子类,通过它能够将捕获的数据输出到QuickTime视频文件里(MOV)

AVCaptureVideoPreviewLayer,是CALayer子类,能够使用它来显示录制的视频

AVCaptureConnection,捕获连接,在一个捕获会话中输入和输出之间的连接

#import "ViewController.h"

#import <AVFoundation/AVFoundation.h>
#import <AssetsLibrary/AssetsLibrary.h> @interface ViewController ()
<AVCaptureFileOutputRecordingDelegate>
{
BOOL isRecording;
} @property (weak, nonatomic) IBOutlet UILabel *label;
@property (weak, nonatomic) IBOutlet UIButton *button; @property (strong, nonatomic) AVCaptureSession *session;
@property (strong, nonatomic) AVCaptureMovieFileOutput *output; - (IBAction)recordPressed:(id)sender; @end @implementation ViewController - (void)viewDidLoad
{
[super viewDidLoad]; self.session = [[AVCaptureSession alloc] init];
self.session.sessionPreset = AVCaptureSessionPresetMedium; AVCaptureDevice *cameraDevice = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo]; NSError *error = nil;
AVCaptureDeviceInput *camera = [AVCaptureDeviceInput deviceInputWithDevice:cameraDevice error:&error]; AVCaptureDevice *micDevice = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeAudio];
AVCaptureDeviceInput *mic = [AVCaptureDeviceInput deviceInputWithDevice:micDevice error:&error]; if (error || !camera || !mic) {
NSLog(@"Input Error");
} else {
//增加捕获音频视频
[self.session addInput:camera];
[self.session addInput:mic];
} self.output = [[AVCaptureMovieFileOutput alloc] init]; if ([self.session canAddOutput:self.output]) {
[self.session addOutput:self.output];//输出
} AVCaptureVideoPreviewLayer *previewLayer = [AVCaptureVideoPreviewLayer layerWithSession:self.session];
previewLayer.frame = CGRectMake(0,0, self.view.frame.size.width, self.view.frame.size.height);
[self.view.layer insertSublayer:previewLayer atIndex:0]; [self.session startRunning];
isRecording = NO;
self.label.text = @""; } - (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
} - (void)viewWillAppear:(BOOL)animated
{
[super viewWillAppear:animated];
if (![self.session isRunning])
{
[self.session startRunning];
}
} - (void)viewWillDisappear:(BOOL)animated
{
[super viewWillDisappear:animated];
if ([self.session isRunning])
{
[self.session stopRunning];
}
} - (IBAction)recordPressed:(id)sender {
if (!isRecording)
{
[self.button setTitle:@"停止" forState:UIControlStateNormal];
self.label.text = @"录制中...";
isRecording = YES;
NSURL *fileURL = [self fileURL];
[self.output startRecordingToOutputFileURL:fileURL recordingDelegate:self];
}
else
{
[self.button setTitle:@"录制" forState:UIControlStateNormal];
self.label.text = @"停止";
[self.output stopRecording];
isRecording = NO;
}
} - (NSURL *) fileURL
{
NSString *outputPath = [[NSString alloc] initWithFormat:@"%@%@", NSTemporaryDirectory(), @"movie.mov"];
NSURL *outputURL = [[NSURL alloc] initFileURLWithPath:outputPath]; NSFileManager *manager = [[NSFileManager alloc] init];
if ([manager fileExistsAtPath:outputPath])
{
[manager removeItemAtPath:outputPath error:nil];
} return outputURL;
} #pragma mark-- AVCaptureFileOutputRecordingDelegate托付协议实现方法 - (void)captureOutput:(AVCaptureFileOutput *)captureOutput
didFinishRecordingToOutputFileAtURL:(NSURL *)outputFileURL
fromConnections:(NSArray *)connections error:(NSError *)error
{ if (error == nil) {
ALAssetsLibrary *library = [[ALAssetsLibrary alloc] init]; [library writeVideoAtPathToSavedPhotosAlbum:outputFileURL
completionBlock:^(NSURL *assetURL, NSError *error)
{
if (error)
{
NSLog(@"写入错误。") ;
} }];
} }

使用UIVideoEditorController

#import "ViewController.h"

@interface ViewController ()
<UIVideoEditorControllerDelegate,UINavigationControllerDelegate> - (IBAction)editButtonPress:(id)sender; @end @implementation ViewController - (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
} - (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
} - (IBAction)editButtonPress:(id)sender { NSBundle *bundle = [NSBundle mainBundle];
NSString *moviePath = [bundle pathForResource:@"YY"
ofType:@"mp4"]; //推断设备是否支持编辑视频
if ([UIVideoEditorController canEditVideoAtPath:moviePath]){ UIVideoEditorController *videoEditor =
[[UIVideoEditorController alloc] init]; videoEditor.delegate = self;
videoEditor.videoPath = moviePath; [self presentViewController:videoEditor animated:YES completion:NULL]; } else {
NSLog(@"不能编辑这个视频");
} } - (void)videoEditorController:(UIVideoEditorController *)editor
didSaveEditedVideoToPath:(NSString *)editedVideoPath{ [editor dismissViewControllerAnimated:YES completion:NULL]; if ( UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(editedVideoPath) ) {
UISaveVideoAtPathToSavedPhotosAlbum(editedVideoPath, self,
@selector(video:didFinishSavingWithError:contextInfo:),
(__bridge void *)(editedVideoPath));
}
} - (void)videoEditorController:(UIVideoEditorController *)editor
didFailWithError:(NSError *)error{
NSLog(@"编辑视频出错");
NSLog(@"Video editor error occurred = %@", error);
[editor dismissViewControllerAnimated:YES completion:NULL];
} - (void)videoEditorControllerDidCancel:(UIVideoEditorController *)editor{
NSLog(@"视频编辑取消");
[editor dismissViewControllerAnimated:YES completion:NULL];
} - (void)video:(NSString *)videoPath
didFinishSavingWithError:(NSError *)error
contextInfo:(NSString *)contextInfo { NSString *title; NSString *message;
if (!error) {
title = @"视频保存";
message = @"视频已经保存到设备的相机胶卷中";
} else {
title = @"视频失败";
message = [error description];
}
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:title
message:message
delegate:nil
cancelButtonTitle:@"OK"
otherButtonTitles:nil];
[alert show];
}

个人感言,事实上视频我之前做过,也有类似文章,在国外的一些站点上全然也能找到比这更好的更有技术含量的代码和文章,只是总之,算是一个学习记录吧~

原书:http://item.jd.com/11522516.html

&lt;图形图像,动画,多媒体&gt; 读书笔记 --- 录制与编辑视频的更多相关文章

  1. &lt;图形图像,动画,多媒体&gt; 读书笔记 --- AirPlay

    AirPlay技术是之前一直没有接触过的技术,正好这次做一个笔记 共用: 1.能够通过AirPlay将iOS和MAC设备上的视频或音频输出到高清电视上或高保真音响 2.能够通过AirPlay将iOS和 ...

  2. &lt;图形图像,动画,多媒体&gt; 读书笔记 --- 力学行为特性

    UIKit力学行为包括了:重力(UIGravityBehavior),碰撞(UICollisionBehavior),吸附(UIAttachmentBehavior),推(UIPushBehavior ...

  3. &lt;图形图像,动画,多媒体&gt; 读书笔记 --- 音效

    音频多媒体文件主要是存放音频数据信息,音频文件在录制的过程中把声音信号,通过音频编码,变成音频数字信号保存到某种格式文件里.在播放过程中在对音频文件解码,解码出的信号通过扬声器等设备就能够转成音波.音 ...

  4. iOS 图形图像动画 Core Animation

    //Core Animation #define WeakSelf __weak __typeof(self) weakSelf = self #define StrongSelf __strong ...

  5. 《Java并发编程实战》第九章 图形用户界面应用程序界面 读书笔记

    一.为什么GUI是单线程化 传统的GUI应用程序通常都是单线程的. 1. 在代码的各个位置都须要调用poll方法来获得输入事件(这样的方式将给代码带来极大的混乱) 2. 通过一个"主事件循环 ...

  6. 【python下使用OpenCV实现计算机视觉读书笔记3】读写视频文件

    代码例如以下: import cv2 videoCapture = cv2.VideoCapture('car.avi') fps = videoCapture.get(cv2.cv.CV_CAP_P ...

  7. 关东升的《iOS实战:图形图像、动画和多媒体卷(Swift版)》上市了

    关东升的<iOS实战:图形图像.动画和多媒体卷(Swift版)>上市了 承蒙广大读者的厚爱我的<iOS实战:图形图像.动画和多媒体卷(Swift版)>京东上市了,欢迎广大读者提 ...

  8. WPF,Silverlight与XAML读书笔记第三十九 - 可视化效果之3D图形

    原文:WPF,Silverlight与XAML读书笔记第三十九 - 可视化效果之3D图形 说明:本系列基本上是<WPF揭秘>的读书笔记.在结构安排与文章内容上参照<WPF揭秘> ...

  9. WPF,Silverlight与XAML读书笔记第四十三 - 多媒体支持之文本与文档

    说明:本系列基本上是<WPF揭秘>的读书笔记.在结构安排与文章内容上参照<WPF揭秘>的编排,对内容进行了总结并加入一些个人理解. Glyphs对象(WPF,Silverlig ...

随机推荐

  1. Drainage Ditches(网络流(EK算法))

    计算最大流,EK算法模板题. #include <stdio.h> #include <string.h> #include <queue> using names ...

  2. ecshop数据库说明

    数据库 ecshop 表的结构 ecs_account_log 字段 类型 空 默认 含义 log_id mediumint(8) 否 账户记录表 user_id mediumint(8) 否 用户编 ...

  3. LeetCode Weekly Contest 28

    1. 551. Student Attendance Record I 2. 552. Student Attendance Record II hihocode原题,https://hihocode ...

  4. MTD 门店统计

    DROP TABLE #MTD ' ,@endDate date = cast(getdate() as date) CREATE TABLE #MTD(bydate date) DECLARE @c ...

  5. P1796 汤姆斯的天堂梦_NOI导刊2010提高(05)

    题目描述 汤姆斯生活在一个等级为0的星球上.那里的环境极其恶劣,每天12小时的工作和成堆的垃圾让人忍无可忍.他向往着等级为N的星球上天堂般的生活. 有一些航班将人从低等级的星球送上高一级的星球,有时需 ...

  6. VMWare 在物理机待机后,报错“该虚拟机似乎正在使用中”

    在物理机待机后,刚打开虚拟机,就弹出这个画面(这种情况经常出现在远程之后,本机待机之后) 点击确定后,就弹出 当点击取消,无反应,而且再次点击VM2又弹出以上窗口,点击获取所有权,则弹出以下窗口 上网 ...

  7. O​r​a​c​l​e​1​1​g​自​带​的​S​Q​L​ ​d​e​v​e​l​o​p​e​r​无​法​打​开​解​决​

    在安装完Oracle Database 11g Release 2数据库,想试一下Oracle自带的SQL Developer工具,在操作系统菜单的所有程序中找到SQL Developer如下所示,并 ...

  8. SLAM: Orb_SLAM中的ORB特征

    原文链接:什么是ORB 关于Orb特征的获取:参考 最新版的OpenCV中新增加的ORB特征的使用 ORB是是ORiented Brief 的简称,对Brief的特定性质进行了改进. ORB的描述在下 ...

  9. POJ_2594_最小路径覆盖

    Treasure Exploration Time Limit: 6000MS   Memory Limit: 65536K Total Submissions: 8085   Accepted: 3 ...

  10. MySql数据库多表操作

    一.连接查询[连表查询.多表查询] 当查询结果的列来源于多张表时,需要将多张表连接成一个大的数据集,再选择合适的列返回 mysql支持三种类型的连接查询,分别为: 内连接查询(inner join) ...