高仿喜马拉雅FM(第一弹)

项目地址github:https://github.com/Eastwu5788/XMLYFM 如果您觉得不错,记得给一个star
高仿喜马拉雅FM(第二弹) 本篇文章有点长了,更多的内容在第二弹继续更新

最新用空闲时间写了一下喜马拉雅FM这款APP。

干货效果演示

多音频下载、本地播放功能演示


LocalDownload.gif

网络音频播放效果演示


PlayDetail.gif

效果演示

推荐页面效果


recom.gif

分类页面效果


cate.gif

广播页面效果


radio.gif

榜单页面效果


Rank.gif

主播页面效果


Anchor.gif

订阅听页面效果


Dingyue.gif

下载听页面效果


xiazai.gif

我的页面效果


Mine.gif

分析

  • 发现tab中有五个小分类,分别对应五个页面,所有在“发现”的控制器中使用了UIPageViewController来控制五个子控制器。
    +从Charles抓出来的接口来看,“推荐”页面一共调用了三个接口,分别请求了推荐、热门、直播的内容,所以在这里选择了Reactivecocoa来实现接口的并发访问
- (void)refreshDataSource {       

    @weakify(self);
    RACSignal *signalRecommend = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        @strongify(self);
        [self requestRecommendList:^{
            [subscriber sendNext:nil];
        }];
        return nil;
    }];    

    RACSignal *signalHotAndGuess = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        @strongify(self);
        [self requestHotAndGuessList:^{
            [subscriber sendNext:nil];
        }];
        return nil;
    }];

    RACSignal *signalLiving = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        @strongify(self);
        [self requestLiving:^{
            [subscriber sendNext:nil];
        }];
        return nil;
    }];

    [[RACSignal combineLatest:@[signalRecommend,signalHotAndGuess,signalLiving]] subscribeNext:^(id x) {
        @strongify(self);
        [(RACSubject *)self.updateContentSignal sendNext:nil];
    }];
}
  • 在“推荐”页面中有几个轮播图,仔细观察会发现它的轮播图一直想左转换,所以这里的轮播图片需要做一下特殊处理。以实现无限轮播的效果
- (void)setModel:(XMLYFindFocusImagesModel *)model  {      
    _model = model;    
    [self.adverScrollView removeAllSubViews];    
    self.adverScrollView.contentSize = CGSizeMake(kScreenWidth * _model.list.count, 150);     
    //1.向scrollView中增加UIImageView的时候,需要在最后一张图片后面将第一张图片添加上去    
    for(NSInteger index = 0; index <= _model.list.count; index++)   {      
        //2.如果是最后一张图片,则放置第一张图片
        XMLYFindFocusImageDetailModel \*detail = index == _model.list.count ? _model.list.firstObject : [_model.list objectAtIndex:index];
        UIImageView \*imageView = [[UIImageView alloc] init];
        imageView.frame = CGRectMake(kScreenWidth \* index, 0, kScreenWidth, 150);
        [imageView yy_setImageWithURL:[NSURL URLWithString:detail.pic] options:YYWebImageOptionSetImageWithFadeAnimation];
        [self.adverScrollView addSubview:imageView];
    }
}
  • 在轮播图滚动动画结束后需要做一下判断,如果当前滚动到了最后一张图片,则立即将scrollView的偏移调整到初始位置,这样一个无限轮播就完成了。
- (void)scrollViewDidScroll:(UIScrollView \*)scrollView {   
    NSInteger curPage = self.adverScrollView.contentOffset.x / kScreenWidth;    
    if(curPage == self.model.list.count) {    
        [self.adverScrollView setContentOffset:CGPointMake(0, 0) animated:NO];    
      }
}
  • 在有轮播图的地方肯定少不了定时器,如果将定时器直接放在cell中,就会因为cell的复用导致定时器出现问题,所有一般是将定时器放在控制器中。但是这样的话也带来一个问题,就是由于定时器的存在,如果要求定时器的生命周期和控制器相同(也就是在控制器dealloc的时候才取消定时器).这样的控制器是无法调用dealloc的,会造成控制器虽然已经退出但是定时器依然在正常工作。所以这里专门为控制器设计了一个定时器的单例帮助类,这样的话就可以在dealloc中去销毁所有的定时器。
@interface XMLYFindRecommendHelper : NSObject    
#pragma mark - Common    
//生成帮助类单例
+ (instancetype)helper;    

//销毁所有的定时器    
- (void)destoryAllTimer;

#pragma mark - Live    

//  开启为直播设置的定时器      
- (void)startLiveTimer;    

//销毁直播的定时器    
- (void)destoryLiveTimer;    

#pragma mark - Header    

//开启头部的定时器    
- (void)startHeadTimer;    

//销毁头部的定时器    
- (void)destoryHeaderTimer;    
@end
  • 在广播页面中,有一个根据当前时间显示不同的问候语的小功能。比如现在是早上6点钟,应该显示“早安*北京”。这里就需要用到NSDateFormatter,但是NSDateFormatter的比较消耗性能,所以我专门写了一个XMLYTimeHelper类来管理所有的时间转换操作。在这个类中对NSDateFormatter做了缓存处理,并使用dispatch_semaphore_t保证了线程安全。
//根据字符串生成相应的NSDateFormatter,比如"yyyy-MM-dd HH:mm:ss"
static force_inline NSDateFormatter *XMLYDataCreateFormatter(NSString *string) {
    NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
    formatter.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US_POSIX"];
    formatter.dateFormat = string;
    return formatter;
}

//用户直接调用此方法,传入"yyyy-MM-dd HH:mm:ss"这样的字符串生成NSDateFormatter
static force_inline NSDateFormatter *XMLYDateFormatter(NSString *string) {
    //1.检查输入的合法性
    if(!string || ![string isKindOfClass:[NSString class]] || string.length == 0) return nil;
   //2.初始化单例参数
    static CFMutableDictionaryRef cache;
    static dispatch_semaphore_t lock;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        cache = CFDictionaryCreateMutable(CFAllocatorGetDefault(), 0, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
        lock = dispatch_semaphore_create(1);
    });

    //3.加锁
    dispatch_semaphore_wait(lock, DISPATCH_TIME_FOREVER);
    //4.查询当前字符串是否已经存在相应的NSDateformatter
    NSDateFormatter *formatter = CFDictionaryGetValue(cache, (__bridge const void *)(string));
   //5.解锁
    dispatch_semaphore_signal(lock);

   //6.如果缓存中没有,则需要重新生成
    if(!formatter) {
        formatter = XMLYDataCreateFormatter(string);
        //7.重新生成成功,存入缓存
        if(formatter) {
            dispatch_semaphore_wait(lock, DISPATCH_TIME_FOREVER);
            CFDictionarySetValue(cache, (__bridge const void *)(string), (__bridge const void *)(formatter));
            dispatch_semaphore_signal(lock);
        }
    }
    return formatter;
}
  • 2016.09.09这次主要是完成了榜单页面和主播页面,榜单页面没有什么特别的东西,主播页面主要使用UICollectionView实现三个cell等分整个屏幕,以实现每个section里面的自动布局。如果想实现无边界的布局,需要重写一下系统的UICollectionViewFlowLayout布局类,否则总是会有一小块边界被显示出来
- (NSArray<UICollectionViewLayoutAttributes *> *)layoutAttributesForElementsInRect:(CGRect)rect {
    //解决issue
    NSArray* attributes = [[NSArray alloc] initWithArray:[super layoutAttributesForElementsInRect:rect] copyItems:YES];
    for(NSInteger i = 1,max = attributes.count; i < max; i++) {
        UICollectionViewLayoutAttributes *currentLayoutAttributes = attributes[i];
        UICollectionViewLayoutAttributes *prevLayoutAttributes = attributes[i - 1];
        NSInteger maximumSpacing = 0;
        NSInteger origin = CGRectGetMaxX(prevLayoutAttributes.frame);
        if(origin + maximumSpacing + currentLayoutAttributes.frame.size.width < self.collectionViewContentSize.width) {
            CGRect frame = currentLayoutAttributes.frame;
            frame.origin.x = origin + maximumSpacing;
            currentLayoutAttributes.frame = frame;
        }
    }
    return attributes;
}

注意,在获取父类的layoutAttributes数组的时候一定要选择copy,否则会报一个issue

2016-09-09 10:20:10.687 XMLYFM[1453:240776] Logging only once for UICollectionViewFlowLayout cache mismatched frame
2016-09-09 10:20:10.688 XMLYFM[1453:240776] UICollectionViewFlowLayout has cached frame mismatch for index path <NSIndexPath: 0xc000000000200116> {length = 2, path = 1 - 1} - cached value: {{106, 415}, {106.66666666666667, 162.53968253968256}}; expected value: {{106.5, 415}, {106.66666666666667, 162.53968253968256}}
2016-09-09 10:20:10.688 XMLYFM[1453:240776] This is likely occurring because the flow layout subclass XMLYAnchorFlowLayout is modifying attributes returned by UICollectionViewFlowLayout without copying them
  • 在我的github上有我一年多以前写的一个通过重写UICollectionViewLayout实现瀑布流的小demo,大家有兴趣可以去看看

  • 在“下载听”页面中有一个显示当前已占用空间和可用空间的功能,关于计算当前可用空间我这里有一个函数,可以直接拿去用。不谢

    static int64_t XMLYDiskSpaceFree() {
      NSError *error = nil;
      NSDictionary *attrs = [[NSFileManager defaultManager] attributesOfFileSystemForPath:NSHomeDirectory() error:&error];
      if (error) return -1;
      int64_t space =  [[attrs objectForKey:NSFileSystemFreeSize] longLongValue];
      if (space < 0) space = -1;
      return space;
    }
  • 在“我的”页面中有一个下拉放大的效果,网上有很多实现的方法,我这里用了一个很简单的方法,在UITableView上增加一个子视图,作为真正的头部视图,遮盖掉tableHeaderView,在ScrollView滚动的时候,改变子视图的大小就行了。

1.创建子视图

- (XMLYMineHeaderView *)headerView {
    if(!_headerView) {
        //真正的头部视图
        _headerView = [[XMLYMineHeaderView alloc] initWithFrame:CGRectMake(0, 0, kScreenWidth, 288)];
        //设置tableHeaderView的大小与头视图相同
        self.tableView.tableHeaderView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, kScreenWidth, 288)];
        //将头部视图作为UITableView子视图,遮盖掉原来的tableHeaderView
        [self.tableView addSubview:_headerView];
    }
    return _headerView;
}

2.TableView滚动时调整headerView的大小

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    CGFloat offsetY = scrollView.contentOffset.y;
    if (offsetY <= 0) {
        self.headerView.frame = CGRectMake(offsetY / 2.0, offsetY, kScreenWidth - offsetY, 288 - offsetY);
    }
}

3.将headerView的frame计算放在layoutSubViews方法中,这样每一次改变headerView的frame,此方法都会走一遍,从而随之调整子视图的位置。但是千万不要把复杂位置计算放在里面

- (void)layoutSubviews { // height 288
    [super layoutSubviews];


    CGFloat hspace = (self.frame.size.width - kScreenWidth) / 2.0f;
    CGFloat centx = self.frame.size.width / 2.0f;

    //背景视图
    self.backImageView.frame = CGRectMake(hspace, 0, kScreenWidth, self.frame.size.height);
    self.alphaView.frame = CGRectMake(hspace, 0, kScreenWidth, self.frame.size.height);

    //节目管理
    self.managerButton.frame = CGRectMake(centx - 10 - 104.0f, self.frame.size.height - 36.0 - 37.0f, 104.0f, 37.0f);

    //录音按钮
    self.recordButton.frame = CGRectMake(centx + 10, self.managerButton.frame.origin.y, 104.0f, 37.0f);

    //子标题 
    self.subTitleLabel.frame = CGRectMake(centx - 150.0f, self.recordButton.frame.origin.y - 24.0f - 15.0f, 300, 15);

    //点击登录按钮
    self.userNameButton.frame = CGRectMake(centx - 100.0f, self.subTitleLabel.frame.origin.y - 10 - 18.0, 200.0f, 18.0f);

    self.avatarImageView.frame = CGRectMake(centx - 45.0, self.userNameButton.frame.origin.y - 10 - 90.0, 90, 90);

    //设置按钮
    self.settingButton.frame = CGRectMake(12 + hspace, self.avatarImageView.frame.origin.y - 20, 20, 20);

}

欢迎大家关注我的微信公众号


qrcode_for_gh_c3cd7518dd64_1280.jpg
作者HansRove,源码XiMaLaYa-by-HansRove-,仿做喜马拉雅, 对AVFoundation框架的一次尝试 软件环境: iOS9.1硬件环境: Mac OS X 10.11开发工具: Xcode7.1 项目描述: 模板是“喜马拉雅FM”,因为很喜欢这个软件的风格和内容,当时这款软件参杂着太多的广告以及推广。所以打算通过抓包,并Json解析出数据,进而使用自己搭建的界面完成视听播放功能, UI也算是高仿喜马拉雅FM.但部分内容在原来基础上做了相应的修改。 功能概述: 0、音频播放:这是最基础的模块,提供用户收听各类声音、专辑及电台主播。 1、发现听:实时动态的信息展示页,每天的音频热门信息和音频专辑集数的更新;展示给用户声音分类和电台及网络主播相应的推荐页供用户选择交互。 2、定制听:记录用户关注的声音(专辑)、主播和电台以及收听历史,并推荐近几天热门声音。 3、下载听:记录用户下载记录并对用户下载过的声音进行分类展示。 4、搜索功能:根据热词提供展示,用户可以搜索主播、电台、声音或专辑。 项目所用技术及框架:【纯代码+故事板】项目使用MVVM模式搭建 1、二次封装AFNetworking与MJExtension进行数据请求与解析; 2、使用Masonry实现纯代码布局,使用MJRefresh实现下拉刷新上拉加载; 3、大量使用自定义Cell、自定义Button、自定义View来实现多控件封装方便布局使用; 4、使用AVFundation实现在线音频播放。并自定义播放器外观以及进度条配置; 项目收获: 1、对MVVM模式的使用更加熟练,设计模式的了解及掌握为未来开发少走了许多弯路; 2、熟悉JSON数据解析,对数据处理有一定的了解,特别是掌握了MJExtension框架的原理及写法; 3、通知中心及单例模式的使用,减低了代码的耦合性; 4、纯代码跳转Storyboard或者Xib使用更加熟练;封装了好多自定义视图,方便布局。
### JavaWeb 实现有声小说平台(类喜马拉雅应用) #### 项目概述 有声小说平台旨在提供在线音频播放服务,允许用户浏览、收听以及管理个人收藏的有声读物。此类应用程序通常具备如下功能模块: - 用户注册与登录验证机制 - 音频资源上传及分类展示 - 搜索栏用于快速查找特定作品 - 收藏夹保存感兴趣的节目列表 - 推荐算法推送个性化内容给听众[^1] #### 技术栈选型 为了构建高效稳定的 Web 应用程序,推荐采用 Spring Boot 框架作为后端开发工具包;前端界面则可以选用 Vue.js 或 React 来增强交互体验。数据库方面建议使用 MySQL 存储结构化数据,Redis 缓存热数据以提高访问速度。 #### 关键组件设计 ##### 后端 API 设计 定义 RESTful APIs 处理客户端请求并返回 JSON 数据响应。以下是几个重要的接口示例: - **获取首页轮播图** ```http GET /api/v1/carousels ``` - **查询所有专辑** ```http GET /api/v1/albums?category={categoryId}&page={pageNum} ``` - **根据 ID 获取单个专辑详情** ```http GET /api/v1/album/{id} ``` - **提交评论** ```http POST /api/v1/comments Content-Type: application/json { "userId": 1, "contentId": 203, "text": "这本小说真不错" } ``` ##### 前端页面布局 利用 HTML5 和 CSS3 构建美观易用的操作面板,确保良好的跨浏览器兼容性和移动端适配效果。对于复杂业务逻辑部分可借助 JavaScript ES6+ 特性简化编码工作量。 ```html <!-- index.html --> <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8"/> <title>有声书城</title> <link rel="stylesheet" href="./assets/css/style.css"/> <script src="./assets/js/app.bundle.js"></script> </head> <body> <div id="app"></div> </body> </html> ``` ##### 客户端 SDK 开发 针对不同操作系统定制专属 App Store 下载链接按钮,方便新老客户下载安装官方移动客户端软件。同时开放第三方开发者接入权限,鼓励更多创新应用场景涌现出来。 #### 示例代码片段 下面给出一段简单的 Spring Boot 控制器方法来处理获取指定编号章节的功能: ```java @RestController @RequestMapping("/chapters") public class ChapterController { @Autowired private IChapterService chapterService; /** * 根据ID查询章节信息. */ @GetMapping("/{chapterId}") public ResponseEntity<ChapterDTO> get(@PathVariable Long chapterId){ try{ Optional<ChapterEntity> optional = chapterService.findById(chapterId); if(!optional.isPresent()){ throw new ResourceNotFoundException("未找到该章节"); } return ResponseEntity.ok(new ChapterDTO(optional.get())); }catch (Exception e){ log.error(e.getMessage(),e); return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build(); } } } ``` 通过上述描述可以看出,在 JavaWeb 平台上创建类似于喜马拉雅这样的大型多媒体分享社区并非难事,只要合理规划架构方案和技术路线即可达成目标。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值