使用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. codevs2596 售货员的难题(状压dp)

    2596 售货员的难题  时间限制: 1 s  空间限制: 32000 KB  题目等级 : 钻石 Diamond     题目描述 Description 某乡有n个村庄(1<n<=15 ...

  2. git add -A (用该命令添加文件时报错)

    前言 最近在学着不使用github客户端,而直接用git命令上传代码至github,当使用命令 git add -A 添加所有项目文件时报错    老实说我是蒙蔽的,因为从来没有遇到过这个错,确认输入 ...

  3. 解决IE不支持Data.parse()的问题

    Date.parse()函数的返回值为Number类型,返回该字符串所表示的日期与 1970 年 1 月 1 日午夜之间相差的毫秒数. var nowtime = "2017-02-08&q ...

  4. MyBatis动态条件、一对多、整合spring(二)

    输入映射和输出映射 Mapper.xml映射文件定义了操作数据库的sql,每一个sql是一个statement,映射文件是mybatis的核心. parameterType输入类型 1.传递简单类型 ...

  5. Django模板常用语法规则

    Django 模板标签 if/else 标签 for 标签 ifequal/ifnotequal 标签 注释标签 过滤器 include 标签 URL标签 模板继承   if/else 标签 1. 基 ...

  6. 软件架构自学笔记——常见的软件架构(https://jiajunhuang.com/articles/2018_09_16-common_software_archtecture_pattern.md.html)

    分层模式 这种模式主要是将设计分层,每一层为其上层提供服务.例如:web开发中我们常常将某些常用的RESTful接口抽象出一个service层. 客户端-服务端模式 客户端和服务端分离,从而解耦.通过 ...

  7. 3.0 Windows和Linux双系统安装(3)

    3.0 Windows和Linux双系统安装(3) 3.1 精简的安装步骤如下:(如果已经有了前面两篇教程的安装经验,推荐看完3.1即可动手了) 双系统很多开发新人会用到,而且比起虚拟机好处是运行效率 ...

  8. android黑科技系列——应用市场省流量更新(增量升级)原理解析

    一.前言 最近在看热修复相关的框架,之前我们已经看过了阿里的Dexposed和AndFix这两个框架了,不了解的同学可以点击这里进行查看:Dexposed框架原理解析 和 AndFix热修复框架原理解 ...

  9. 关于基础的Set 和Get

    先附上一篇文章,讲的很清楚 在Core中,我们要是先这样设置了.在我们对这个上下文做查询工作的时候,例如: var head = _OMSECDatabase.OmsEcorderHead.Where ...

  10. SSIS 无法在 unicode 和非 unicode 字符串数据类型之间转换

    最近在学SSIS,遇到一个问题,把平面文件源的数据导入到EXCEL中. 平面文件源的对象是CSV,读进来的PhoneNumber是 DT_STR 然后倒入Excel 对应列建立的是longtext 一 ...