iOS 编程:视频背景视图&图片视差效果

视频背景视图

实现方式一:使用WKWebView

使用 WKWebView 实现视频背景视图:

  1. WKWebView 直接加载 gif 图片;
  2. WKWebView 加载 HTML 页面,HTML页面中包含 gif 图片;

使用方法2: 用 HTML 页面加载 gif 图片而不是直接加载 gif 图片的原因是,在 HTML 中可以通过 CSS 将图片全屏显示,还可以保持图片的原始比例。

STEP 1: 使用 GIF Brewery 将视频文件转换为 gif 图片

STEP 2: 制作HTML页面

WebViewContent.html

<!DOCTYPE html>
<html style="height: 100%;">
<head>
    <meta charset="utf-8">
    <title>含有GIF的web页面</title>
</head>
<body style="margin:0; height: 100%">
    <div style="background: url(behance.gif) center center / cover no-repeat; height: 100%" ></div>
</body>
</html>

STEP 3: 使用 WKWebView 加载并显示HTML页面,并且将 WKWebView 作为背景视图:

- (void)viewDidLoad {
    [super viewDidLoad];
        
    // 加载含有 gif 图片的 HTML 文件
    NSString *htmlPath = [[NSBundle mainBundle] pathForResource:@"WebViewContent" ofType:@"html"];
    NSURL *htmlURL = [[NSURL alloc] initFileURLWithPath:htmlPath];
    NSData *htmlData = [[NSData alloc] initWithContentsOfURL:htmlURL];
    
    // 使用 WKWebView 视图加载并显示 gif 图片
    WKWebView *webView = [[WKWebView alloc] initWithFrame:self.view.frame];
    [webView loadData:htmlData MIMEType:@"text/html" characterEncodingName:@"UTF-8" baseURL:[htmlURL URLByDeletingLastPathComponent]];
    webView.userInteractionEnabled = NO;
    [self.view addSubview:webView];
    
    // 遮罩层视图
    UIView *filter = [[UIView alloc] initWithFrame:self.view.frame];
    filter.backgroundColor = [UIColor blackColor];
    filter.alpha = 0.05;
    [self.view addSubview:filter];
}

实现方式二:使用YYImage

YYImage 是一个功能强大的 iOS 图像框架,它支持 WebP, PNG, GIF, JPEG, JP2, TIFF, BMP, ICO, ICNS 等图片的播放/编码/解码操作。

显示动画类型的图片:

UIImage *image = [YYImage imageNamed:@"railway.gif"];
UIImageView *imageView = [[YYAnimatedImageView alloc] initWithImage:image];
imageView.frame = self.view.bounds;
imageView.contentMode = UIViewContentModeScaleAspectFill;
[self.view addSubview:imageView];

实现方式三:使用 AVFoundation

实现步骤:

  1. AVPlayerItem 加载视频文件;
  2. AVPlayer 用于播放 AVPlayerItem,并进行一些播放设置,设置播放完成后自动重新播放的通知;
  3. 将播放图层 AVPlayerLayer 添加到当前视图图层上;
  4. 应用从前台切换到后台时会停止播放视频,因此,当应用从后台恢复到前台状态时,需要恢复播放视频,需要设置一个全局的通知。
#import "ViewController.h"

// 1.需要导入 AVFoundation 框架
#import <AVFoundation/AVFoundation.h>

@interface ViewController ()

/** 全屏播放器 */
@property (nonatomic, strong) AVPlayer *player;

@end

@implementation ViewController

#pragma mark - Lifecycle

- (void)viewDidLoad {
    [super viewDidLoad];

    [self setupForAVplayerView];
}

- (void)viewWillAppear:(BOOL)animated {
    // 视频播放通知
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(playVideos)
                                                 name:@"videoshouldplay"
                                               object:nil];
}

#pragma mark - Custom Accessors

- (AVPlayer *)player {
    if (!_player) {
        
        // AVPlayerItem
        NSString *filePath = [[NSBundle mainBundle] pathForResource:@"BridgeLoop-640p" ofType:@"mp4"];
        NSURL *url = [NSURL fileURLWithPath:filePath];
        AVPlayerItem *playerItem = [AVPlayerItem playerItemWithURL:url];
        
        // AVPlayer
        _player = [AVPlayer playerWithPlayerItem:playerItem];
        // 设置重复播放
        _player.actionAtItemEnd = AVPlayerActionAtItemEndNone;
        // 播放结束时的通知
        [[NSNotificationCenter defaultCenter]addObserver:self
                                                selector:@selector(playerItemDidPlayToEndTimeNotification:)
                                                    name:AVPlayerItemDidPlayToEndTimeNotification
                                                object:nil];
    }
    return _player;
}

#pragma mark - Private

- (void)setupForAVplayerView {
    //创建播放图层
    AVPlayerLayer *playerLayer = [AVPlayerLayer playerLayerWithPlayer:self.player];
    playerLayer.frame = self.view.bounds;
    [self.view.layer addSublayer:playerLayer];
}

- (void)playVideos {
    [self.player play];
}

// 播放结束通知
- (void)playerItemDidPlayToEndTimeNotification:(NSNotification *)sender {
    [_player seekToTime:kCMTimeZero]; // 播放结束后,设置重新播放
}

@end

当应用在前后台之间切换时,播放会暂停,因此每当应用进入前台时,需要发送通知:

// AppDelegate.m
- (void)applicationDidBecomeActive:(UIApplication *)application {
    // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.
    [[NSNotificationCenter defaultCenter] postNotificationName:@"videoshouldplay" object:nil];
}

图片视差效果

iOS用户界面有一个动态效果,背景图片会随着手机的水平晃动而跟随晃动。

打开/关闭此功能:设置—通用—辅助功能—减弱动态效果。

如果你在iOS中安装了一个叫做 红板报 的新闻类应用,那么你应该注意到,它的「封面故事」中的图片也有视差效果:

红板报:手机沿各个方向倾斜的视差效果

为图片添加视差效果很简单,以下是如何实现此功能的示例代码:

// 添加图片
UIImage *image = [UIImage imageNamed:@"albume.png"];
UIImageView *imageView = [[UIImageView alloc] initWithImage:image];
imageView.frame = self.view.bounds;
imageView.contentMode = UIViewContentModeScaleAspectFill;
[self.view addSubview:imageView];

// 图片视差效果:水平方向
UIInterpolatingMotionEffect *effectX = [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.x" type:UIInterpolatingMotionEffectTypeTiltAlongHorizontalAxis];
effectX.maximumRelativeValue = @(-50);
effectX.minimumRelativeValue = @(50);
[imageView addMotionEffect:effectX];

// 图片视差效果:垂直方向
UIInterpolatingMotionEffect *effectY = [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.y" type:UIInterpolatingMotionEffectTypeTiltAlongVerticalAxis];
effectY.maximumRelativeValue = @(-50);
effectY.minimumRelativeValue = @(50);
[imageView addMotionEffect:effectY];

参考:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容