项目中常用功能,如:流媒体、健康数据(步数等)等-b

整理iOS开发中使用的各种流媒体和常用的高级功能。由于时间关系,目前只写了一部分功能,全部都采用的是系统方法,没用第三方,截图如下:

screen1.png

screen2.png

个人比较懒,不爱多写文字,直接上代码,哈哈。

视频

系统用AVFoundation与MediaPlayer框架实现播放视频的方案。其中AVFoundation扩展性好,都需自定义功能,而MediaPlayer集成简单,但是样式不可扩展。

1.AVFoundation使用AVPlayer播放视频,它属于view的layer层。其功能都需要自定义,如音量、暂停、播放时长等。代码如下:

- (void)addAVPlayer{    
    self.playerItem = [AVPlayerItem playerItemWithURL:self.movieURL];    self.player = [AVPlayer playerWithPlayerItem:self.playerItem];    AVPlayerLayer *layer = [AVPlayerLayer playerLayerWithPlayer:self.player];    
    layer.frame = CGRectMake(0, 70, kScreenWidth, 300);    
    layer.videoGravity = AVLayerVideoGravityResizeAspect;    
    layer.backgroundColor = [[UIColor blackColor] CGColor];    
    [self.view.layer addSublayer:layer];    
    [self.player play];
}
#pragma mark - Observer//添加进度观察
- (void)addProgressObserver {    
    //  设置每秒执行一次    
    __weak LXAVPlayVideoController *wSelf = self;    
    [self.player addPeriodicTimeObserverForInterval:CMTimeMake(1, 1) queue: NULL usingBlock:^(CMTime time) {        //        NSLog(@"进度观察 + %f", wSelf.topProgressSlider.value);         //  获取当前时间        
        CMTime currentTime = wSelf.player.currentItem.currentTime;        //  转化成秒数        
        CGFloat currentPlayTime = (CGFloat)currentTime.value/currentTime.timescale;        //  总时间        
        CMTime totalTime = wSelf.playerItem.duration;        //  转化成秒        
        wSelf.totalMovieDuration = (CGFloat)totalTime.value/totalTime.timescale;          
        wSelf.topProgressSlider.value = currentPlayTime/wSelf.totalMovieDuration;        
        wSelf.progressValue = CMTimeGetSeconds(currentTime)/wSelf.totalMovieDuration;        
        wSelf.topPastTimeLabel.text = [LXHelpClass getTimeByProgress:currentPlayTime];        
        wSelf.topRemainderLabel.text = [LXHelpClass getTimeByProgress:wSelf.totalMovieDuration - currentPlayTime];        //        NSLog(@"%f %f %f %f", wSelf.topProgressSlider.value, wSelf.totalMovieDuration, currentPlayTime, currentPlayTime/wSelf.totalMovieDuration);    
   }];
}
//播放结束后的代理回调

- (void)moviePlayDidEnd:(NSNotification *)notify {    
    [self setMovieParse];
}
#pragma mark - Enent response//播放进度
- (void)topSliderValueChangedAction:(UISlider *)sender {    
    NSLog(@"进度条进度 + %f", sender.value);    
    double currentTime = floor(self.totalMovieDuration * sender.value);    //转换成CMTime才能给player来控制播放进度    
    CMTime dragedCMTime = CMTimeMake(currentTime, 1);    
    [self.player seekToTime:dragedCMTime completionHandler:^(BOOL finished) {        
        [self.player play];    
    }];
}
//音量slider

- (void)bottomSoundSliderAction:(UISlider *)sender {    
    [self.player setVolume:sender.value];    
    self.bottomSoundSlider.value = sender.value;    
    if (sender.value == 0) {        
        NSLog(@"静音");    
    }
}
//播放暂停按钮
- (void)playBtnClicked:(UIButton *)sender {    
    if (isPlay) {        
        [self setMovieParse];    
    } else {        
        [self setMoviePlay];    
    }      
    isPlay ^= 1;
}
#pragma mark - Private methods

- (void)addNotificationCenters {    
    //  注册观察者用来观察,是否播放完毕    
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(moviePlayDidEnd:) name:AVPlayerItemDidPlayToEndTimeNotification object:nil];
}  
- (void)setMovieParse {    
    [self.player pause];
}  
- (void)setMoviePlay {    
    [self.player play];
}

2.MediaPlayer框架提供MPMoviePlayerController与MPMoviePlayerViewController播 放视频,它们区别就是MPMoviePlayerViewController里面包含了一个MPMoviePlayerController,另外 MPMoviePlayerViewController可以看作是一个控制器播放视频的。系统已经为之集成好了音量、播放暂停等功能。

  • MPMoviePlayerController的使用
- (void)addMPMoviePlayer{    
    MPMoviePlayerController *mpPlayer = [[MPMoviePlayerController alloc] init];    
    mpPlayer.view.frame = CGRectMake(0, 20, kScreenWidth, kScreenWidth/videoSizeRate);    
    mpPlayer.backgroundView.backgroundColor = [UIColor blackColor];    
    mpPlayer.scalingMode = MPMovieScalingModeAspectFill;    
    [self.view addSubview:mpPlayer.view];    
    self.mpPlayer = mpPlayer;
}
#pragma mark - Observer

- (void)enterFullscreen:(NSNotification *)noti {    
    [self fullScreen:YES];    
    [LXHelpClass setDeviceLandscape:direction];
}
- (void)exitFullscreen:(NSNotification *)noti {    
    [self fullScreen:NO];    
    UIDeviceOrientation orient = [UIDevice currentDevice].orientation;    
    if (orient != UIDeviceOrientationPortrait) {        
        direction = UIDeviceOrientationPortrait;        
        [LXHelpClass setDeviceLandscape:direction];     }    
    direction = UIDeviceOrientationLandscapeLeft;
}
- (void)loadCompletion:(NSNotification *)noti {    
    [self.videoLoading stopAnimating];
}
- (void)orientChange:(NSNotification *)noti {    
    UIDeviceOrientation orient = [UIDevice currentDevice].orientation;    
    switch (orient)     {        
        case UIDeviceOrientationPortrait:            
            self.mpPlayer.fullscreen = NO;            
            break;        
        case UIDeviceOrientationLandscapeLeft:            
            direction = UIDeviceOrientationLandscapeLeft;            
            self.mpPlayer.fullscreen = YES;            
            break;        
        case UIDeviceOrientationPortraitUpsideDown:            
            self.mpPlayer.fullscreen = NO;            
            break;        
        case UIDeviceOrientationLandscapeRight:            
            direction = UIDeviceOrientationLandscapeRight;            
            self.mpPlayer.fullscreen = YES;            
            break;        
            default:  
            break;    
     }
}  - (void)mediaPlayerPlaybackStateChange:(NSNotification *)notification {    
    switch (self.mpPlayer.playbackState) {        
        case MPMoviePlaybackStatePlaying:            
            NSLog(@"正在播放...");            
            break;        
        case MPMoviePlaybackStatePaused:            
            NSLog(@"暂停播放.");//注意播放完成时的状态是暂停            
            break;        
        case MPMoviePlaybackStateStopped:            
            NSLog(@"停止播放.");            
            break;        
        default:            
            NSLog(@"播放状态:%li",self.mpPlayer.playbackState);            
            break;    
      }
}  
- (void)mediaPlayerThumbnailRequestFinished:(NSNotification *)notification {    
    NSLog(@"视频截图完成.");    
    UIImage *image=notification.userInfo[MPMoviePlayerThumbnailImageKey];    //保存图片到相册(首次调用会请求用户获得访问相册权限)    
    UIImageWriteToSavedPhotosAlbum(image, nil, nil, nil);
}
  • MPMoviePlayerViewController的使用
- (void)viewDidLoad {    
    [super viewDidLoad];    
    self.view.backgroundColor = bgColor;    
    MPMoviePlayerViewController *player = [[MPMoviePlayerViewController alloc] initWithContentURL:self.mp4URL];    
    player.view.frame = CGRectMake(0, 0, kScreenWidth, 300);    
    [self.view addSubview:player.view];    
    self.player = player;    
    UIButton * button = [UIButton buttonWithType:UIButtonTypeRoundedRect];    
    button.frame = CGRectMake((kScreenWidth-100)/2.0, kScreenHeight - 64 - 100, 100, 50);    
    [button setTitle:@"全屏播放" forState:UIControlStateNormal];    
    [button addTarget:self action:@selector(buttonClick) forControlEvents:UIControlEventTouchUpInside];    
    [self.view addSubview:button];
}  
- (void)buttonClick {    
    [self presentMoviePlayerViewControllerAnimated:self.player];
}

音频

系统也提供灵活性很强的AVAudioPlayer和集成快的MPMusicPlayerController播放音乐。
具体功能实现请查看源码,这里不做过多介绍。

健康

这方面的资料比较少,只好查看官方文档。这里我只写了获取步数这个常用功能。HealthKit框架提供了许多获取健康数据的API。

LXHealthKitManager *myHealth = [LXHealthKitManager sharedLXHealthKitManager];
myHealth.startDate = [LXHealthKitManager getTodayAgoWithDays:30];//获取今天之前三十天的步数[myHealth getStepCountWithSuccess:^(NSArray *results) {    
[self.modelArr setArray:results];    
dispatch_async(dispatch_get_main_queue(), ^{        
    [self.tableView reloadData];    
});
} withfailure:^(NSError *error) {    
    NSLog(@"error:%@", error);
}];

LXHealthKitManager的代码如下:

- (instancetype)init {    
    self = [super init];    
    if (self) {        
        [self setup];    
    }    
    return self;
}  
- (void)setup {    
    if ([HKHealthStore isHealthDataAvailable]) {        
        NSLog(@"此设备能使用健康数据");        
        HKHealthStore *healthStore = [[HKHealthStore alloc] init];        
        [healthStore requestAuthorizationToShareTypes:[self shareTypes] readTypes:[self readTypes] completion:^(BOOL success, NSError *error) {            
        if (success == YES)  {                
            NSLog(@"授权成功");            
        } else {                
            NSLog(@"授权失败");            
        }     }];        
    self.healthStore = healthStore;    
    } else {        
        NSLog(@"此设备不能使用健康数据");    
    }
}  
- (void)getStepCountWithSuccess:(void (^)(NSArray *))success withfailure:(void (^)(NSError *))failure {    
    HKQuantityType *quantityType = [HKQuantityType quantityTypeForIdentifier:HKQuantityTypeIdentifierStepCount];    NSPredicate *predicate = [HKQuery predicateForSamplesWithStartDate:self.startDate endDate:self.endDate options:HKQueryOptionStrictEndDate];    NSDateComponents *dateComponents = [[NSDateComponents alloc] init];    
    dateComponents.day = 1;      
    HKStatisticsCollectionQuery *collectionQuery = [[HKStatisticsCollectionQuery alloc] initWithQuantityType:quantityType quantitySamplePredicate:predicate options:HKStatisticsOptionCumulativeSum | HKStatisticsOptionSeparateBySource anchorDate:self.startDate intervalComponents:dateComponents];      
    collectionQuery.initialResultsHandler = ^(HKStatisticsCollectionQuery *query, HKStatisticsCollection * __nullable result, NSError * __nullable error) {        NSMutableArray *returnArr = [NSMutableArray array];        for (NSInteger i = result.statistics.count - 1; i >= 0; i--) {            
        HKStatistics *statistic = result.statistics[i];            for (HKSource *source in statistic.sources) {                if ([source.name isEqualToString:[UIDevice currentDevice].name]) {                    
            LXHealthStepModel *model = [[LXHealthStepModel alloc] init];                    
            model.startDateStr = [self changeToDateStrWithDate:statistic.startDate];                    
            model.endDateStr = [self changeToDateStrWithDate:statistic.endDate];                      
            HKQuantity *quantity = [statistic sumQuantityForSource:source];                    if ([quantity isCompatibleWithUnit:[HKUnit countUnit]]) {                        
            model.stepCount = (NSInteger)[quantity doubleValueForUnit:[HKUnit countUnit]];                     

            }                    
            [returnArr addObject:model];                
            }            
          }        
      }        
      if (!error) {            
            if (success) {                
                  success(returnArr);            
            }        
       } else if (failure) {            
               failure(error);        
       }    
    };    
    [self.healthStore executeQuery:collectionQuery];
}

源码请点击github地址:https://github.com/SoftProgramLX/LXBaseFunction

简书作者:App开发LX

地址:http://www.jianshu.com/p/c62886e0ac3c

时间: 2024-05-30 11:25:48

项目中常用功能,如:流媒体、健康数据(步数等)等-b的相关文章

项目中使用的ajax异步读取数据结构设计

设计稍微复杂了一点,完成ajax读取功能涉及到了很多页面.虽然如此,但感觉比较灵活. 和传统方法唯一的区别在于多了一层数据容器控件,里面提供了显示数据的HTML元素及相应的JS方法. 这样数据控件指生成纯数据. ajax异步读取 使用了jQuery.ajax,通过ajax POST方式请求后台处理ashx页面,并传递相关参数. ashx 完成动态加载用户控件,并根据接收的参数对控件的属性进行赋值. 加载控件,借助于博客园老赵的一篇博文,链接找不到了,以后再补. public class View

re模块中常用功能函数

re模块中常用功能函数 正则表达式是一个特殊的字符序列,它能帮助你方便的检查一个字符串是否与某种模式匹配. Python 自1.5版本起增加了re 模块,它提供 Perl 风格的正则表达式模式. re 模块使 Python 语言拥有全部的正则表达式功能. compile 函数根据一个模式字符串和可选的标志参数生成一个正则表达式对象.该对象拥有一系列方法用于正则表达式匹配和替换. re 模块也提供了与这些方法功能完全一致的函数,这些函数使用一个模式字符串做为它们的第一个参数. re.match函数

项目中常用的SQL语句

摘要: 对于像我一样的菜鸟级别人物来说,接触的项目比较少,平常做的那些练习,包括三层中,一般也只能用到增删改查,当然这些增删改查语句是学数据库语句的入门,在百度上搜索一大堆,w3cschool中帮助文档也可以查阅. 这里是把平常用的增删改做一个小小总结,对一些小小项目也是比较适用的 ,当然,好记性不如烂笔头吗,这个时代纸质版的笔记感觉很欠缺,因为想要看的时候,你得找到那本书,而且要翻翻在哪里,感觉很不方便.并且电子版 的随便copy之类的都可以. 常用的sql语句(增删改查) 1.增加.插入语句

C#项目中常用到的设计模式

1. 引言 一个项目的通常都是从Demo开始,不断为项目添加新的功能以及重构,也许刚开始的时候代码显得非常凌乱,毫无设计可言.但是随着项目的迭代,往往需要将很多相同功能的代码抽取出来,这也是设计模式的开始.熟练运用设计模式应该是每一个软件开发人员的必备技能.今天给大家介绍几个常用的设计模式. 2. 单例模式 单例模式恐怕是很多开发人员最先接触到的模式之一,可以认为就是一个全局变量.它的初始化过程无非就是一开始就new 一个instance,或者惰性初始化等需要用到的时候new 一个instanc

iOS项目中常用的第三方开源库

1.项目使用的第三方开源库 http://github.ibireme.com/github/list/ios/整理了比较常用的iOS第三方组件,以及github上的统计. 项目使用了CocoaPods(类似java中的maven)管理常用的第三方库,一些特殊的单独引用,下面介绍下比较好用的几个. (1)AFNetworking 目前比较推荐的iOS网络请求组件,默认网络请求是异步,通过block回调的方式对返回数据进行处理. 需要注意的是AFNetworking对服务器返回的ContentTy

Swift 项目中常用的第三方框架

Swift 项目中可能用到的第三方框架 字数1004 阅读4091 评论17 喜欢93 这里记录下swift开发中可能用的框架 , 最近浏览了不少,积累在这里,以后用的时候方便查阅.顺便推荐给大家! 这里的框架都是纯swift的 , 感谢开源 ,感谢大神们 . 下拉刷新 BreakOutToRefresh 一个下拉刷新打砖块的swift库 SDRefreshView 简单易用的上拉和下拉刷新 ZLSwiftRefresh - 下拉刷新/上拉加载更多,支持自定义动画,集成简单 GearRefres

简析项目中常用的七参数转换法和四参数转换法以及涉及到的基本测量学知识

1.背景 在了解这两种转换方法时,我们有必要先了解一些与此相关的基本知识.我们有三种常用的方式来表示空间坐标,分别是:经纬度和高层.平面坐标和高层以及空间直角坐标. 2.经纬度坐标系(大地坐标系) 这里我首先要强调:天文坐标表示的经纬度和大地坐标系表示的经纬度是不同的.所以,同一个经纬度数值,在BJ54和WGS84下表示的是不同的位置,而以下我说的经纬度均指大地坐标系下的经纬度.大地坐标系是大地测量中以参考椭球面为基准面建立起来的坐标系.下面我跟大家大致谈谈其中涉及到的两个重要概念. 2.1大地

项目总结二:关于项目中一些功能的优化

前言: 最近一直在维护老的项目,遇到的问题也千奇百怪,需要修补的,需要优化的,需要特殊处理的,感觉总是那么的无语.也许这时候也应该感叹一句:路漫漫其修远兮,吾将上下而求索吧. 这篇文章就只是讲讲老项目中遇到的种种不敢苟同的代码写法,以及遇到一些问题时的处理方法. 1.关于按钮事件的重复点击问题 我们开发中大概都经历过这样的事情,我的一个button被重复的快速点击,(我们开发者应该更清楚的知道这意味着什么).那我们再深入的想一下(给这样的事件安排一个特定的环境):例如当网络较差的情况下,再例如当

项目中常用SQL语句总结

1.项目中常常需要修改字段长度,但需要保留数据--增加业务受理 项目名称 字段长度alter table t_ywsl add aa varchar2(200);update t_ywsl set aa=proname,proname=null;commit;alter table t_ywsl modify proname varchar2(200);update t_ywsl set proname=aa,aa=null;commit;alter table t_ywsl drop colu