1. 前言

什么是故障艺术(Glitch Art 风)?我们熟知的抖音的 LOGO 正是故障艺术其中一种表现形式。它有一种魔幻的感觉,看起来具有闪烁、震动的效果,很吸引人眼球。故障艺术它模拟了画面信号出现故障导致成像错误的感觉。青色色块与红色色块无法重合就是这种故障的体现。从胶片时代开始到今天的数码时代,这种故障一直是观众非常熟悉的现象。

上个月看到CSS 故障艺术这篇文章,最近想转换心情于是开始抄它的动画了(顺便为博客园的UWP板块吊命)。CSS的mix-blend-mode好像很好用,这次用UWP中Win2D的BlendEffect模仿它的玩法。

2. 什么是BlendEffect

Microsoft.Graphics.Canvas.Effects命名空间下的BlendEffect 用于组合两张图片(分别是作为输入源的Background和Foreground),它包含多种模式,如下图所示:

要使用BlendEffect,需要为它设置Mode,并将Foreground和Background设置为CompositionEffectSourceParameter

var blendEffect = new BlendEffect()
{
Mode = BlendEffectMode.Screen,
Foreground = new CompositionEffectSourceParameter("Main"),
Background = new CompositionEffectSourceParameter("Tint"),
};

然后使用这个BlendEffect创建Brush,并用SetSourceParameter设置它的Foreground和Background为需要混合的两个CompositionBrush

var effectFactory = compositor.CreateEffectFactory(blendEffect);
var blendEffectBrush = effectFactory.CreateBrush();
blendEffectBrush.SetSourceParameter("Main", foregroundBrush);
blendEffectBrush.SetSourceParameter("Tint", backgroundBrush);

这样Foreground和Background就会根据Mode指定的模式进行混合。

3. 各种模式下的文字混合效果

CompositionAPI没有提供文字书写的功能,倒是Win2D有提供这个功能,我在用Win2D实现镂空文字有介绍如何使用Win2D创建出一个包含文字的CompositionSurfaceBrush,这次把它重新封装了一下,然后使用下面的代码显示各种模式下的混合效果。

private void AddTextToRoot(BlendEffectMode blendEffectMode)
{
var redBrushWrapper = CreateTextToBrushWrapper(blendEffectMode.ToString(), Colors.Red);
var blueBrushWrapper = CreateTextToBrushWrapper(blendEffectMode.ToString(), Colors.Cyan);
blueBrushWrapper.Brush.Offset = new Vector2(-4f, 0); var textVisual = Compositor.CreateSpriteVisual();
textVisual.Brush = CreateBrush(blueBrushWrapper.Brush, redBrushWrapper.Brush, blendEffectMode);
textVisual.Size = new Vector2(400, 70);
var background = new Rectangle { Height = 70, Width = 400 }; ElementCompositionPreview.SetElementChildVisual(background, textVisual);
Root.Children.Add(background);
} private CompositionBrush CreateBrush(CompositionBrush foreground, CompositionBrush background, BlendEffectMode blendEffectMode)
{
var compositor = Window.Current.Compositor;
var effect = new BlendEffect()
{
Mode = blendEffectMode,
Foreground = new CompositionEffectSourceParameter("Main"),
Background = new CompositionEffectSourceParameter("Tint"),
};
var effectFactory = compositor.CreateEffectFactory(effect);
var compositionBrush = effectFactory.CreateBrush();
compositionBrush.SetSourceParameter("Main", foreground);
compositionBrush.SetSourceParameter("Tint", background); return compositionBrush;
}

上图为各种BlendEffectMode下红色和蓝色文字的混合效果。

4. 图片的 Glitch Art 风

理解BlendEffect的用法和作用后,现在可以做些故障艺术的尝试了。拿下面这张图片来尝试:

private (CompositionBrush compositionBrush, CompositionSurfaceBrush compositionSurfaceBrush) CreateBrush(string imageName, Color color)
{
var compositor = Window.Current.Compositor;
var loadedSurface = LoadedImageSurface.StartLoadFromUri(new Uri("ms-appx:///Assets/Images/" + imageName));
var compositionSurfaceBrush = compositor.CreateSurfaceBrush();
compositionSurfaceBrush.Surface = loadedSurface;
var compositionBrush = CreateBrush(compositionSurfaceBrush, compositor.CreateColorBrush(color), BlendEffectMode.Lighten);
return (compositionBrush, compositionSurfaceBrush);
}

使用上面的代码分别将这张图片与从Colors.Red(255,0,0)和Colors.Cyan(0,255,255)创建出来的CompositionColorBrush使用BlendEffectMode.Lighten进行混合,得到下面的效果:

var compositor = Window.Current.Compositor;
var (foreground, foregroundBrush) = CreateBrush(imageName, Colors.Cyan);
var (background, backgroundBrush) = CreateBrush(imageName, Colors.Red);
foregroundBrush.Offset = new Vector2(10, 0); var brush = CreateBrush(foreground, background, BlendEffectMode.Darken);

使用BlendEffectMode.Darken将两张图片混合,再把其中一张图偏移10像素后,得到如下效果,这样图片的故障艺术分格就完成了:

5. 动态类抖音风格 Glitch 效果

继续抄动画,连标题一起抄。将两个文字的CompositionSurfaceBrush使用BlendEffectMode.Lighten进行混合,然后对它们的Offset做动画:

CreateBrush(backgroundWrapper.Brush, foregroundWrapper.Brush, BlendEffectMode.Lighten);

StartOffsetAnimation(backgroundWrapper.Brush, TimeSpan.FromSeconds(0.95), TimeSpan.Zero);
StartOffsetAnimation(foregroundWrapper.Brush, TimeSpan.FromSeconds(1.1), TimeSpan.FromSeconds(0.2)); private void StartOffsetAnimation(CompositionSurfaceBrush brush, TimeSpan duration, TimeSpan delay)
{
var offsetAnimation = Compositor.CreateVector2KeyFrameAnimation();
offsetAnimation.Duration = duration;
offsetAnimation.DelayTime = delay;
offsetAnimation.IterationBehavior = AnimationIterationBehavior.Forever; void addKey(float key, float top, float left)
{
offsetAnimation.InsertKeyFrame(key, new Vector2(top * 2.5f, left * 2.5f));
};
addKey(.1f, -0.4f, -1.1f);
addKey(.2f, 0.4f, -0.2f);
addKey(.3f, 0f, .5f);
addKey(.4f, -0.3f, -0.7f);
addKey(.5f, 0, .2f);
addKey(.6f, 1.8f, 1.2f);
addKey(.7f, -1f, .1f);
addKey(.8f, -0.4f, -0.9f);
addKey(.9f, 0, 1.2f);
addKey(1, 0, -1.2f);
brush.StartAnimation(nameof(CompositionSurfaceBrush.Offset), offsetAnimation);
}

再在中间放一条上下移动的黑线,故障的效果就出来了:

6. 更复杂的Glitch效果

这次没法完全抄CSS的动画了,我能力有限,大致抄个意思得了。

创建两个白色黑底带阴影(BlurAmount = 0)的文字(如下所示):

错开几个像素后用BlendEffectMode.Multiply混合在一起,一种更复杂的故障艺术动画就完成了:

再在后面放一个白色的文字,分别调整前面两个文字的高度,效果如下:

然后对前面两个文字的高度进行动画,效果就出来了:

StartHeightAnimation(redBrushWrapper, new List<(double, double)>() { (0, 1), (20, 80), (60, 15), (100, 105) }, TimeSpan.FromSeconds(1), TimeSpan.Zero);
StartHeightAnimation(blueBrushWrapper, new List<(double, double)>() { (0, 110), (20, 112.5), (35, 30), (50, 100), (60, 50), (70, 85), (80, 55), (100, 1) }, TimeSpan.FromSeconds(1.5), TimeSpan.Zero); private void StartHeightAnimation(TextToBrushWrapper brush, List<(double, double)> keyFrames, TimeSpan duration, TimeSpan delay)
{
var storyboard = new Storyboard(); var animation = new DoubleAnimationUsingKeyFrames();
animation.EnableDependentAnimation = true;
Storyboard.SetTarget(animation, brush);
Storyboard.SetTargetProperty(animation, nameof(TextToBrushWrapper.Height)); foreach (var item in keyFrames)
{
animation.KeyFrames.Add(new LinearDoubleKeyFrame { KeyTime = duration / 100 * item.Item1, Value = item.Item2 });
} storyboard.Children.Add(animation);
storyboard.RepeatBehavior = RepeatBehavior.Forever; storyboard.BeginTime = delay;
storyboard.Begin();
}

7. 结语

为了从大佬那里抄动画我还特地新建了一个项目

原文还有很多动画可以参考,但我半途而废了,搞明白大致的原理后新鲜感就过去了。看起来CPU、GPU的占用也不高,只是暂时想不到有什么实际应用的场景,博客也写了,图片也截了,玩腻就抛弃。

8. 源码

DinoChan_uwp_design_and_animation_lab 一个收集UWP的设计和动画的项目。

[UWP]抄抄《CSS 故障艺术》的动画的更多相关文章

  1. [UWP]使用AlphaMaskEffect提升故障艺术动画的性能(顺便介绍怎么使用性能探测器分析UWP程序)

    前几天发布了抄抄<CSS 故障艺术>的动画这篇文章,在这篇文章里介绍了如何使用Win2D绘制文字然后配合BlendEffect制作故障艺术的动画.本来打算就这样收手不玩这个动画了,但后来又 ...

  2. CSS 故障艺术

    本文的主题是 Glitch Art,故障艺术. 什么是故障艺术?我们熟知的抖音的 LOGO 正是故障艺术其中一种表现形式.它有一种魔幻的感觉,看起来具有闪烁.震动的效果,很吸引人眼球. 故障艺术它模拟 ...

  3. 抖音抖一抖-SVG和CSS视觉故障艺术小赏

    故障艺术,英文名称叫glitch,在很多赛博朋克作品中经常看到,其实就是故意表现一种显示设备的小故障效果,抖音的图标其实就是这种的效果,我们看下这个图标 这个图标中的红色和蓝色的偏移其实就是一种故障艺 ...

  4. 有趣的 CSS 像素艺术

    原文地址:https://css-tricks.com/fun-times-css-pixel-art/#article-header-id-4 译者:nzbin 友情提示:由于国内网络的原因,Cod ...

  5. css的艺术

    鲁先生曾经说过:"html和css都不能算一门语言..." html确实不能算一门语言,他只是二三十个英语单词而已,但是css不一样,css是==艺术== 骚操作 上图的12个图标 ...

  6. 纯css实现苹果表盘动画

    欢迎訪问我们的博客:http://www.w3ctrain.com/2015/07/06/Apple-Watch-Dials/ 随着苹果表的大量生产,我想.用CSS来实现拨号动画的时候到了. 在这篇文 ...

  7. No.3 - CSS transition 和 CSS transform 配合制作动画

    课程概述 作业提交截止时间:09-01 任务目的 深度理解掌握 transition-timing-function 以及它的意义 学会配合使用 CSS transform 和CSS transiti ...

  8. uwp - 做一个相对炫酷的动画按钮/按钮动画

    原文:uwp - 做一个相对炫酷的动画按钮/按钮动画 看腻了系统自带的button animation何不尝试下自定义一个较为炫酷的动画顺便提升用户体验.效果图: 动画分为几个部分,分别是:内圆从中心 ...

  9. css吃豆人动画

    一. Css吃豆人动画 1. 上半圆:两个div,内部一个圆div,外部设置宽高截取半圆 外部div动画:animation: 动画样式 1s(时长) ease(动画先低速后快速) infinite( ...

随机推荐

  1. 《JavaScript算法》二分查找的思路与代码实现

    二分查找的思路 首先,从有序数组的中间的元素开始搜索,如果该元素正好是目标元素(即要查找的元素),则搜索过程结束,否则进行下一步. 如果目标元素大于或者小于中间元素,则在数组大于或小于中间元素的那一半 ...

  2. B站实战第三天

    B站实战第三天 用了两天多的时间才把B站页面的头部写完,今天来写头部下面的导航栏部分和轮播图一些模块. 因为还没学js,轮播图部分用swiper来实现. 今天首先复习的知识点是弹性盒模型. 弹性盒模型 ...

  3. spring——AOP原理及源码(一)

    教程共分为五篇,从AOP实例的构建及其重要组件.基本运行流程.容器创建流程.关键方法调用.原理总结归纳等几个方面一步步走进AOP的世界. 本篇主要为读者演示构建AOP实例及AOP核心组件分析. 一.项 ...

  4. Java面试必问之Hashmap底层实现原理(JDK1.7)

    1. 前言 Hashmap可以说是Java面试必问的,一般的面试题会问: Hashmap有哪些特性? Hashmap底层实现原理(get\put\resize) Hashmap怎么解决hash冲突? ...

  5. Deepin环境下启动Pycharm没有启动图标解决办法

    小伙伴们在deepin下运行pycharm时,是不是需要通过sh文件启动? 下面告诉大家如何将pycharm图标放在桌面上: 1.在桌面打开终端,输入命令: sudo gedit /usr/share ...

  6. Java 读取Word中的脚注、尾注

    本文介绍读取Word中的脚注及尾注的方法,添加脚注.尾注可以参考这篇文章. 注:本文使用了Word类库(Free Spire.Doc for Java 免费版)来读取,获取该类库可通过官网下载,并解压 ...

  7. 写react项目要注意的事项

    1,className一定是大写字母开头,例如:App-logo,App,App-header. 2,有关react元素的更新,唯一办法是创建新元素,然后重新将其传入ReactDOM.render() ...

  8. vue中犯下的小错误(一)

    在开发采筑平台SRM的移动项目中:一个页面,感觉没啥错误,但是页面报错如下: 页面中的data或者mothods都没有任何问题,但是这个报错很是让人纠结,后来发现,在使用子组件时候: 此tabShow ...

  9. 机器学习实战:意大利Covid-19病毒感染数学模型及预测

    作者:Gianluca Malato deephub翻译组:刘欣然 当今世界正在与一个新的敌人作斗争,那就是Covid-19病毒. 该病毒自首次在中国出现以来,在世界范围内迅速传播.不幸的是,意大利的 ...

  10. 第四篇(1):企业常用Linux web环境安装配置(apache、php、mysql)

    上篇我们讲了基本的软件包管理和文件操作什么的,现在也要动手安装点有用的东西了吧! 本篇我会写出一个用yum安装apache.php.mysql的方法,最后再运行phpMyAdmin来管理数据库. 1. ...