Metal与图形渲染七:蓝线挑战

零. 前言

蓝线挑战,曾经一度风靡各大短视频平台的一个玩法,不乏有大神利用这个玩法产生了一系列的神作,更不乏失败踩雷的各种捧腹之作,今天我们来用Metal实现一下这个可玩性很高的挑战吧~

一. 原理概述

蓝线挑战的特点是:处于蓝线扫过的地方,取的是之前渲染过的内容;处于蓝线未扫过的地方,取的是当前摄像头的内容,而处于蓝线的范围,取的自然是蓝线的色值,于是乎我们得到一条渲染链:

摄像头获取到CVPixelBuffer后,让MovieReader处理生成纹理,BlueLineFilter先根据摄像头的纹理和自己上一帧处理过的输出纹理渲染进行,然后让DrawBlueLineFilter进行蓝线的绘制,最终渲染到RenderView上面去。

核心就是:处理好上一帧之后的纹理要存储好,和当前摄像头的纹理进行渲染,就可以得到合起来的内容啦~

二. BlueLineFilter

该Filter核心是如何存储之前渲染好的内容和获取当前的内容进行渲染,其Shader如下:

fragment float4 blueLineFragment(TwoInputVertexIO input [[ stage_in ]],
                                 texture2d<float> cameraTexture [[texture(0)]],
                                 texture2d<float> screenShotTexture [[texture(1)]],
                                 constant float &offset [[ buffer(0) ]],
                                 constant bool &isVertical [[ buffer(1) ]])
{
    constexpr sampler quadSampler;
    float4 cameraColor = cameraTexture.sample(quadSampler, input.textureCoordinate);
    float4 screenShotColor = screenShotTexture.sample(quadSampler, input.textureCoordinate2);
    
    float coor = isVertical ? input.position.y : input.position.x;
    
    if (coor < offset) {
        return screenShotColor;
    } else {
        return cameraColor;
    }
}

代码逻辑非常简单,Offset代表当前蓝线处于的位置,如果处于蓝线之前,则取上一帧的输出,如果处于蓝线之后,则取当前摄像头的输出。水平移动取x,垂直移动取y。

来看看怎么获取上一帧的纹理的:

- (instancetype)initWithRenderContext:(HobenMetalRenderContext *)renderContext {
    return [super initWithVertexName:@"twoInputVertex" fragmentName:@"blueLineFragment" numberOfInputs:2 renderContext:renderContext];
}

- (void)newTextureAvailable:(id<MTLTexture>)texture index:(NSInteger)index commandBuffer:(id<MTLCommandBuffer>)commandBuffer {
    [super newTextureAvailable:texture index:index commandBuffer:commandBuffer];
    
    if (!_lastTexture) {
        _lastTexture = texture;
    }

    [super newTextureAvailable:_lastTexture index:1 commandBuffer:commandBuffer];
}

- (void)renderToTextureWithVertices:(NSArray *)vertices textureCoordinates:(NSArray *)textureCoordinates {
    for (HobenMetalTexture *inputTexture in _inputTextures) {
        inputTexture.textureCoordinates = textureCoordinates;
    }
    float offset = _percent * _lastTexture.height;
    id <MTLBuffer> offsetBuffer = [_renderContext.device newBufferWithBytes:&offset length:sizeof(float) options:MTLResourceStorageModeShared];
    bool isVertical = _isVertical;
    id <MTLBuffer> isVerticalBuffer = [_renderContext.device newBufferWithBytes:&isVertical length:sizeof(bool) options:MTLResourceStorageModeShared];
    
    [_renderContext renderQuad:_pipelineState inputTextures:_inputTextures imageVertices:vertices vertexBuffers:nil fragmentBuffers:@[offsetBuffer, isVerticalBuffer] outputTexture:_outputTexture commandBuffer:_filterCommandBuffer];
    
    [self transmitTextureToAllTargets:_outputTexture commandBuffer:_filterCommandBuffer];
    
    self.lastTexture = _outputTexture;
}

同样比较清晰,先声明这是个双输入Filter,然后根据存储上一帧的outputTexture作为输入,和当前摄像头的纹理进行渲染即可,percent和isVertical都是由外部定义好传进来的。

三. DrawBlueLineFilter

该Filter主要作用是进行蓝线的绘制,Shader如下:

constant float4 blueLineColor = float4(0, 1, 1, 1);

constant float blueLineSize = 5;

fragment float4 drawBlueLineFragment(SingleInputVertexIO input [[ stage_in ]],
                                     texture2d<float> inputTexture [[texture(0)]],
                                     constant float &offset [[ buffer(0) ]],
                                     constant bool &isVertical [[ buffer(1) ]])
{
    constexpr sampler quadSampler;
    float4 inputTextureColor = inputTexture.sample(quadSampler, input.textureCoordinate);
    
    float coor = isVertical ? input.position.y : input.position.x;
    
    if (coor < offset || coor > offset + blueLineSize) {
        return inputTextureColor;
    } else {
        return blueLineColor;
    }
}

原理其实和上面差不多,到底是取蓝线颜色还是取摄像头颜色,逻辑很清晰了,这里就不讲解了~

四. 外部调用

其实就是加个定时器和蓝线移动开关,也没啥好说的。

- (void)startTimer {
    _percent = 0;
    [self stopTimer];
    __weak typeof(self) weakSelf = self;
    self.timer = [NSTimer timerWithTimeInterval:0.015 repeats:YES block:^(NSTimer * _Nonnull timer) {
        weakSelf.percent += 0.001;
    }];
    
    [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
}

- (void)stopTimer {
    if (self.timer) {
        [self.timer invalidate];
        self.timer = nil;
    }
}

- (void)setPercent:(CGFloat)percent {
    _percent = percent;
    
    if (percent > 1) {
        [self stopRecord];
        return;
    }
    
    self.blueLineFilter.percent = percent;
    self.drawBlueLineFilter.percent = percent;
}

五. 总结

这是我做过最好玩的一个Demo,趣味性非常高,能搞笑也能秀操作。这个项目主要的难点是如何获取到上一帧的纹理,但因为自己封装了一个可复用性较高的MetalKit,所以也不算特别难,越来越感觉到链式渲染的好用了~

参考:使用OpenGL挑战抖音蓝线特效

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

推荐阅读更多精彩内容