JavaScript Canvas实现噪点滤镜回忆童年电视雪花屏

 更新时间:2022年09月19日 09:20:30   作者:jsmask  
这篇文章主要为大家介绍了JavaScript Canvas实现噪点滤镜回忆童年电视雪花屏,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

相信很多人80,90后的同学对童年里电视机的突然出现刺啦刺啦的雪花屏记忆犹新,本期将用 pixi.js 来完成一个电视机播放动漫然后突然出现雪花屏的动画,里面主要讲解了如何使用pixi.js播放帧动画和如何用噪点滤镜制造雪花屏。

正文

初始化渲染

import * as PIXI from "pixi.js"
const GAME_WIDTH = 800;
const GAME_HEIGHT = 600;
export default class Game {
    constructor(el) {
        this.isOpen = false;
        return this.init(el)
    }
    init(el) {
        this.app = new PIXI.Application({
            width: GAME_WIDTH,
            height: GAME_HEIGHT,
            backgroundColor: 0x000000,
            antialias: true,
            clearBeforeRender: true,
            antialias: true,
            autoDensity: true,
            resolution: window.devicePixelRatio || 1,
        });
        el.appendChild(this.app.view);
        this.loader = new PIXI.Loader();
        this.loader
            .add("tv", "./assets/SnowflakeScreen/TV.png")
            .load(this.render.bind(this))
        return this;
    }
    render(loader, resources) {
        // 渲染
        this.resources = resources; 
        // 创建容器
        this.container = new PIXI.Container();
        this.container.interactive = true;
        this.container.buttonMode = true;
        this.app.stage.addChild(this.container);
        this.tv = this.drawTV();
        this.container.addChild(this.tv);
    }
    drawTV() {
        // 绘制电视机
        let tv = PIXI.Sprite.from(this.resources.tv.texture);
        tv.anchor.set(0.5);
        tv.position.set(GAME_WIDTH / 2, GAME_HEIGHT / 2)
        tv.scale.set(.7, .7)
        tv.zIndex = 9;
        return tv;
    }
}

初始化就是创建 pixi.js 应用,然后把生成后的视图追加到要传入的 el 元素节点上。这里我们还要先加载一张电视机png格式的图片。加载完这张图后,我们才会执行 render 方法进行渲染。先创建主容器 container 并添加到场景舞台中,同时开启 interactive 为了可以给他下面的元素设置 zIndex 来改变图层优先级。 然后就用刚加载的电视机图片的纹理图变成图片精灵添加到主容器中,就这样电视机就绘制到屏幕中央了。

雪花屏动画

实现雪花屏动画之前先来绘制一个电视屏幕:

export default class Game {
	render(loader, resources) {
        // 渲染
        // ...
        this.noise = this.drawNoise()
        this.noise.visible = true
        this.container.addChild(this.noise);
    }
    drawNoise() {
        let noise = new PIXI.Graphics()
        noise.beginFill(0xffffff, .55)
        noise.drawRect(105, 100, 450, 360)
        noise.zIndex = 1
        return noise;
    }
}

其实就是一个简单的白色透明矩形矩形放置在电视机之下。

然后,接下来就是在这个半透明矩形上用噪点滤镜,就可以出现雪花效果:

this.noise.filters = [new PIXI.filters.NoiseFilter(.6, Math.random())]

里面可以传两个值,第一个值代表了噪声强度应为范围[0,1],第二个值则是用于噪声生成的随机量,感兴趣的可以看它具体的 shader 实现

当然,现在画面虽然是雪花,但是它还不会动,非常简单,我们只要在每次渲染时改变它就好了。

export default class Game {
	render(loader, resources) {
        // 渲染
        // ...
        this.noise.filters = []
        this.app.ticker.add(this.step.bind(this));
    }   
    step() {
        this.noise.filters = [new PIXI.filters.NoiseFilter(.6, Math.random())]
    }
}

帧动画播放

先要准备好播放的帧动画图片:

export default class Game {
	render(loader, resources) {
        // 渲染
        // ...
       	this.video = this.drawVideo()
        this.container.addChild(this.video);      
    }    
	drawVideo() {
        let textureList = []
        for (let i = 1; i <= 28; i++) {
            let texture = PIXI.Texture.from(`assets/SnowflakeScreen/video/ezgif-frame-${(i + "").padStart(3, 0)}.jpg`);
            textureList.push(texture);
        };
        const video = new PIXI.AnimatedSprite(textureList);
        video.anchor.set(.5, .5)
        video.position.set(GAME_WIDTH / 2, GAME_HEIGHT / 2 - 10)
        video.scale.set(.48, .48)
        video.loop = false;
        video.animationSpeed = .16
        return video;
    }
}

我们把这28张图片先生成纹理图,按照顺序保存到一个数组里面,然后可以把这个数组传入一个实例化的 AnimatedSprite 类中,里面可以控制它的大小位置是否循环或者播放速度等信息,添加到容器里就可以自动播放啦。

export default class Game {
	render(loader, resources) {
        // 渲染
        // ...
        this.video.visible = false;
        this.video.onComplete = () => {
            this.video.visible = false;
            this.noise.visible = true
        }
        this.container.on('pointerdown', (e) => {
            if(!this.isOpen){
                this.isOpen = true;
                this.video.visible = true;
                this.noise.visible = false
                this.video.gotoAndPlay(0)
            }
            else{
                this.isOpen = false;
                this.noise.filters.length = 0
                this.video.visible = false;
                this.noise.visible = true;
            }
        })
    }
    step() {
        if(!this.isOpen) return;
        this.noise.filters = [new PIXI.filters.NoiseFilter(.6, Math.random())]
    }
}

最后加入一些逻辑比如播放完开启雪花屏,容器点击后播放动画等,非常快速的就可以实现这个电视机播放动漫然后突然出现雪花屏的动画效果了。

以上就是JavaScript Canvas实现噪点滤镜回忆童年电视雪花屏的详细内容,更多关于JavaScript Canvas 噪点滤镜的资料请关注脚本之家其它相关文章!

相关文章

  • 带你理解JavaScript 原型原型链

    带你理解JavaScript 原型原型链

    理解js中原型、原型链这个概念,绝对是帮助我们更深入学习js的必要一步,比如,如果js开发者想理解js继承,new关键字原理,甚至封装组件、优化代码,弄明白js中原型、原型链更是前提条件。本篇文章,用最简洁的文字,清楚明白讲解原型链相等关系和原型、原型链存在的意义
    2021-09-09
  • javascript实现字典Dictionary示例基础

    javascript实现字典Dictionary示例基础

    这篇文章主要为大家介绍了javascript实现字典Dictionary基础示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-08-08
  • 一文详解typeScript的extends关键字

    一文详解typeScript的extends关键字

    这篇文章主要为大家介绍了typeScript的extends关键字使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • Lodash加减乘除add subtract multiply divide方法源码解读

    Lodash加减乘除add subtract multiply divide方法源码解读

    这篇文章主要介绍了Lodash加减乘除add subtract multiply divide方法源码解读,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-05-05
  • 使用xml2js库进行XML数据解析

    使用xml2js库进行XML数据解析

    这篇文章主要为大家介绍了使用xml2js库进行XML数据解析用法详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-09-09
  • JS中轻松遍历对象属性的几种方式

    JS中轻松遍历对象属性的几种方式

    这篇文章主要给大家介绍的是JS中轻松遍历对象属性的几种方式,文章从自身可枚举属性、Object.values() 返回属性值、Object.entries()来展开介绍,感兴趣的小伙伴可以参考一下
    2021-09-09
  • 微信小程序支付之c#后台实现方法

    微信小程序支付之c#后台实现方法

    这篇文章主要介绍了微信小程序支付之c#后台实现方法的相关资料,希望通过本文能帮助到大家,让大家实现这样的功能,需要的朋友可以参考下
    2017-10-10
  • 微信小程序 this和that详解及简单实例

    微信小程序 this和that详解及简单实例

    这篇文章主要介绍了微信小程序 this和that详解及简单实例的相关资料,需要的朋友可以参考下
    2017-02-02
  • js 数组 fill() 填充方法

    js 数组 fill() 填充方法

    这篇文章主要介绍了js 数组 fill() 的填充方法,初始化数组的方法,但是初始化数组之后,数组中的每一项元素默认为 empty 空位占位,如何对数组这些空位添加默认的元素,ES6提供了 fill() 方法实现这一操作。本文总结数组 fill() 方法的详细使用,需要的朋友可以参考一下
    2021-10-10
  • javascript条件式访问属性和箭头函数介绍

    javascript条件式访问属性和箭头函数介绍

    这篇文章主要介绍了javascript条件式访问属性和箭头函数,下面文章围绕条件式访问属性和箭头函数的相关资料展开文章内容,需要的朋友可以参考一下
    2021-11-11

最新评论