使用长按手势与 AVFoundation 开始停止录制
Use long press gesture with AVFoundation to start stop recording
您好,我想通过长按手势在 Objective-C 中开始停止使用 AVFoundation 编码的录制。
我知道如何使用另一个按钮开始录制和停止录制,但不能用同一个按钮完成所有操作。
谢谢!
你可以像这样在里面设置一个UILongPressGestureRecognizer
- (void) viewDidLoad {....}
UILongPressGestureRecognizer *gesture1 = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(longPressed:)];
//[gesture1 setDelegate:self]; you won't need this most probably if detecting the long press is your only expectation from the gesture recognizer
[gesture1 setMinimumPressDuration:1];
[self addGestureRecognizer:gesture1];
//instead of [self addGest... you can try [self.view addGest.. as follows:
[self.view addGestureRecognizer:gesture1];
当用户按下视图至少 1 秒时,将触发此功能:
-(void)longPressed:(UIGestureRecognizer *)longPress {
if(([longPress state] == UIGestureRecognizerStateEnded) || ([longPress state] == UIGestureRecognizerStateEnded)) {
NSLog(@"long press ended");
//[self stopRecording];
}
else if([longPress state] == UIGestureRecognizerStateBegan) {
NSLog(@"long press detected");
//[self startRecording];
}
}
而不是您使用 UIButton 的旧方法:
-(IBAction) buttonPressed: (id)sender {
NSLog(@"button press detected");
//[self startRecording];
}
我在 github 中对您的代码进行了很少的更改,并在下面发布了代码。我添加 UIGestureRecognizerDelegate
的原因是因为长按手势会忽略您的 PLAYBUTTON
的点击。为了避免延迟,我将 [gesture1 setMinimumPressDuration:0];
更改为 0 秒。
当我 运行 项目时,它完全按照你的要求工作:)
这是.h
//viewcontroller.h
#import <UIKit/UIKit.h>
#import <AVFoundation/AVFoundation.h>
#import <CoreAudio/CoreAudioTypes.h>
@interface ViewController : UIViewController <AVAudioRecorderDelegate, AVAudioPlayerDelegate, UIGestureRecognizerDelegate>
@property (weak, nonatomic) IBOutlet UIButton *playButton;
- (void)longPressed:(UIGestureRecognizer *)longPress;
- (IBAction)playTapped:(id)sender;
@end
实现如下:
//viewcontroller.m
#import "ViewController.h"
@interface ViewController ()
{
AVAudioRecorder *recorder;
AVAudioPlayer *player;
}
@end
@implementation ViewController
@synthesize playButton;
- (void)viewDidLoad
{
UILongPressGestureRecognizer *gesture1 = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(longPressed:)];
gesture1.delegate = self;
[gesture1 setMinimumPressDuration:0];
[self.view addGestureRecognizer:gesture1];
[super viewDidLoad];
// Disable Stop/Play button when application launches
[playButton setEnabled:NO];
// Set the audio file
NSArray *pathComponents = [NSArray arrayWithObjects:
[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject],
@"MyAudioMemo.m4a",
nil];
NSURL *outputFileURL = [NSURL fileURLWithPathComponents:pathComponents];
// Setup audio session
AVAudioSession *session = [AVAudioSession sharedInstance];
[session setCategory:AVAudioSessionCategoryPlayAndRecord error:nil];
// Define the recorder setting
NSMutableDictionary *recordSetting = [[NSMutableDictionary alloc] init];
[recordSetting setValue:[NSNumber numberWithInt:kAudioFormatMPEG4AAC] forKey:AVFormatIDKey];
[recordSetting setValue:[NSNumber numberWithFloat:44100.0] forKey:AVSampleRateKey];
[recordSetting setValue:[NSNumber numberWithInt: 2] forKey:AVNumberOfChannelsKey];
// Initiate and prepare the recorder
recorder = [[AVAudioRecorder alloc] initWithURL:outputFileURL settings:recordSetting error:nil];
recorder.delegate = self;
recorder.meteringEnabled = YES;
[recorder prepareToRecord];
}
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
- (void)longPressed:(UIGestureRecognizer *)longPress {
if(([longPress state] == UIGestureRecognizerStateEnded) || ([longPress state] == UIGestureRecognizerStateEnded)) {
NSLog(@"long press ended");
[recorder stop];
AVAudioSession *audioSession = [AVAudioSession sharedInstance];
[audioSession setActive:NO error:nil];
}
else if([longPress state] == UIGestureRecognizerStateBegan) {
NSLog(@"long press detected");
if (player.playing) {
[player stop];
}
if (!recorder.recording) {
AVAudioSession *session = [AVAudioSession sharedInstance];
[session setActive:YES error:nil];
// Start recording
[recorder record];
}
}
}
- (IBAction)playTapped:(id)sender {
NSLog(@"playTapped");
if (!recorder.recording){
player = [[AVAudioPlayer alloc] initWithContentsOfURL:recorder.url error:nil];
[player setDelegate:self];
[player play];
}
}
#pragma mark - AVAudioRecorderDelegate
- (void) audioRecorderDidFinishRecording:(AVAudioRecorder *)avrecorder successfully:(BOOL)flag{
NSLog(@"audioRecorderDidFinishRecording");
[playButton setEnabled:YES];
}
#pragma mark - AVAudioPlayerDelegate
- (void) audioPlayerDidFinishPlaying:(AVAudioPlayer *)player successfully:(BOOL)flag{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle: @"Done"
message: @"Finish playing the recording!"
delegate: nil
cancelButtonTitle:@"OK"
otherButtonTitles:nil];
[alert show];
}
#pragma mark - UIGestureRecognizerDelegate
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch {
// test if our control subview is on-screen
if (playButton.superview != nil) {
if ([touch.view isDescendantOfView:playButton]) {
// we touched our control surface
return NO; // ignore the touch
}
}
return YES; // handle the touch
}
@end
您好,我想通过长按手势在 Objective-C 中开始停止使用 AVFoundation 编码的录制。
我知道如何使用另一个按钮开始录制和停止录制,但不能用同一个按钮完成所有操作。
谢谢!
你可以像这样在里面设置一个UILongPressGestureRecognizer
- (void) viewDidLoad {....}
UILongPressGestureRecognizer *gesture1 = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(longPressed:)];
//[gesture1 setDelegate:self]; you won't need this most probably if detecting the long press is your only expectation from the gesture recognizer
[gesture1 setMinimumPressDuration:1];
[self addGestureRecognizer:gesture1];
//instead of [self addGest... you can try [self.view addGest.. as follows:
[self.view addGestureRecognizer:gesture1];
当用户按下视图至少 1 秒时,将触发此功能:
-(void)longPressed:(UIGestureRecognizer *)longPress {
if(([longPress state] == UIGestureRecognizerStateEnded) || ([longPress state] == UIGestureRecognizerStateEnded)) {
NSLog(@"long press ended");
//[self stopRecording];
}
else if([longPress state] == UIGestureRecognizerStateBegan) {
NSLog(@"long press detected");
//[self startRecording];
}
}
而不是您使用 UIButton 的旧方法:
-(IBAction) buttonPressed: (id)sender {
NSLog(@"button press detected");
//[self startRecording];
}
我在 github 中对您的代码进行了很少的更改,并在下面发布了代码。我添加 UIGestureRecognizerDelegate
的原因是因为长按手势会忽略您的 PLAYBUTTON
的点击。为了避免延迟,我将 [gesture1 setMinimumPressDuration:0];
更改为 0 秒。
当我 运行 项目时,它完全按照你的要求工作:) 这是.h
//viewcontroller.h
#import <UIKit/UIKit.h>
#import <AVFoundation/AVFoundation.h>
#import <CoreAudio/CoreAudioTypes.h>
@interface ViewController : UIViewController <AVAudioRecorderDelegate, AVAudioPlayerDelegate, UIGestureRecognizerDelegate>
@property (weak, nonatomic) IBOutlet UIButton *playButton;
- (void)longPressed:(UIGestureRecognizer *)longPress;
- (IBAction)playTapped:(id)sender;
@end
实现如下:
//viewcontroller.m
#import "ViewController.h"
@interface ViewController ()
{
AVAudioRecorder *recorder;
AVAudioPlayer *player;
}
@end
@implementation ViewController
@synthesize playButton;
- (void)viewDidLoad
{
UILongPressGestureRecognizer *gesture1 = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(longPressed:)];
gesture1.delegate = self;
[gesture1 setMinimumPressDuration:0];
[self.view addGestureRecognizer:gesture1];
[super viewDidLoad];
// Disable Stop/Play button when application launches
[playButton setEnabled:NO];
// Set the audio file
NSArray *pathComponents = [NSArray arrayWithObjects:
[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject],
@"MyAudioMemo.m4a",
nil];
NSURL *outputFileURL = [NSURL fileURLWithPathComponents:pathComponents];
// Setup audio session
AVAudioSession *session = [AVAudioSession sharedInstance];
[session setCategory:AVAudioSessionCategoryPlayAndRecord error:nil];
// Define the recorder setting
NSMutableDictionary *recordSetting = [[NSMutableDictionary alloc] init];
[recordSetting setValue:[NSNumber numberWithInt:kAudioFormatMPEG4AAC] forKey:AVFormatIDKey];
[recordSetting setValue:[NSNumber numberWithFloat:44100.0] forKey:AVSampleRateKey];
[recordSetting setValue:[NSNumber numberWithInt: 2] forKey:AVNumberOfChannelsKey];
// Initiate and prepare the recorder
recorder = [[AVAudioRecorder alloc] initWithURL:outputFileURL settings:recordSetting error:nil];
recorder.delegate = self;
recorder.meteringEnabled = YES;
[recorder prepareToRecord];
}
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
- (void)longPressed:(UIGestureRecognizer *)longPress {
if(([longPress state] == UIGestureRecognizerStateEnded) || ([longPress state] == UIGestureRecognizerStateEnded)) {
NSLog(@"long press ended");
[recorder stop];
AVAudioSession *audioSession = [AVAudioSession sharedInstance];
[audioSession setActive:NO error:nil];
}
else if([longPress state] == UIGestureRecognizerStateBegan) {
NSLog(@"long press detected");
if (player.playing) {
[player stop];
}
if (!recorder.recording) {
AVAudioSession *session = [AVAudioSession sharedInstance];
[session setActive:YES error:nil];
// Start recording
[recorder record];
}
}
}
- (IBAction)playTapped:(id)sender {
NSLog(@"playTapped");
if (!recorder.recording){
player = [[AVAudioPlayer alloc] initWithContentsOfURL:recorder.url error:nil];
[player setDelegate:self];
[player play];
}
}
#pragma mark - AVAudioRecorderDelegate
- (void) audioRecorderDidFinishRecording:(AVAudioRecorder *)avrecorder successfully:(BOOL)flag{
NSLog(@"audioRecorderDidFinishRecording");
[playButton setEnabled:YES];
}
#pragma mark - AVAudioPlayerDelegate
- (void) audioPlayerDidFinishPlaying:(AVAudioPlayer *)player successfully:(BOOL)flag{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle: @"Done"
message: @"Finish playing the recording!"
delegate: nil
cancelButtonTitle:@"OK"
otherButtonTitles:nil];
[alert show];
}
#pragma mark - UIGestureRecognizerDelegate
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch {
// test if our control subview is on-screen
if (playButton.superview != nil) {
if ([touch.view isDescendantOfView:playButton]) {
// we touched our control surface
return NO; // ignore the touch
}
}
return YES; // handle the touch
}
@end