前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Flutter & GLSL - 叁 | 变量传参

Flutter & GLSL - 叁 | 变量传参

作者头像
张风捷特烈
发布2024-02-17 09:08:25
960
发布2024-02-17 09:08:25
举报
Flutter & GLSL 系列文章:
1. 从尺寸入参开始说起

上一篇介绍了,在着色器中坐标和颜色的关系,将坐标归 1 后留下一个问题:

如何让着色器代码中的 size 不写死,由外界传递呢?


在着色器代码中,可以通过 uniform 定义 vec2 类型变量 uSize:

代码语言:javascript
复制
---->[shaders/var_01.frag]----
#version 460 core
precision mediump float;
#include <flutter/runtime_effect.glsl>

uniform vec2 uSize;
out vec4 fragColor;

void main() {
    vec2 coo = FlutterFragCoord().xy/uSize;
    fragColor = vec4(1,0.0,0.0,1.0);
}

Flutter 中通过 FragmentShader#setFloat 传递变量,如下所示:索引 0 表示 uSize 第一个分量,也就是宽度;索引 1 设置高度:

代码语言:javascript
复制
---->[lib/paint/shaders/var_demos/v1_painter.dart]----
class ShaderPainter extends CustomPainter {
  ShaderPainter({required this.shader});

  FragmentShader shader;

  @override
  void paint(Canvas canvas, Size size) {
    shader.setFloat(0, size.width);
    shader.setFloat(1, size.height);
    final paint = Paint()..shader = shader;
    canvas.drawRect(Rect.fromLTWH(0, 0, size.width, size.height), paint,);
  }

  @override
  bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
}

这就是最简单从 Flutter 像 GLSL 传递参数的方式。


2. 选择颜色进行渐变

下面再通过一个案例熟悉一下入参的处理,我们在下方准备了一些备选色,现在的需求是

将选择的颜色作为入参,通过着色器展示 黑色 → 选中色 的渐变效果:

道理是一样的,颜色是一个四维向量 vec4,作为参数传入着色器程序。定义 uniform vec4 uColor; 然后通过 mix 函数将黑色和传入颜色,根据像素的横坐标进行混合。

mix 是一个内置函数,由三个入参 a,b,t 。表示用于在 a, b 个值在 t 分度时的线性混合。 举个小例子:8 和 24 在 0.4 处的混合值是 8 + (24 -8)*0.4 对于多维的值,就是各个分量的混合值。

代码语言:javascript
复制
---->[shaders/var_02.frag]----
#version 460 core
precision mediump float;
#include <flutter/runtime_effect.glsl>

uniform vec2 uSize;
uniform vec4 uColor;
out vec4 fragColor;

void main() {
    vec2 coo = FlutterFragCoord().xy/uSize;
    vec4 black = vec4(0,0.0,0.0,1.0);
    fragColor = mix(black,uColor,coo.x);
}

在 Flutter 中也是通过 setFloat 传入各个分量的值,索引顺序按照GLSL 代码中变量定义的顺序。其中索引 0,1 表示尺寸的二维向量;那么 2,3,4,5 就是表示颜色的四个分量:

代码语言:javascript
复制
---->[lib/paint/shaders/var_demos/v2_painter.dart]----
class V2ShaderPainter extends CustomPainter {
  V2ShaderPainter({required this.shader, required this.color});

  FragmentShader shader;
  Color color;

  @override
  void paint(Canvas canvas, Size size) {
    shader.setFloat(0, size.width);
    shader.setFloat(1, size.height);
    shader.setFloat(2, color.red / 255);
    shader.setFloat(3, color.green / 255);
    shader.setFloat(4, color.blue / 255);
    shader.setFloat(5, color.alpha / 255);
    final paint = Paint()..shader = shader;
    canvas.drawRect(
      Rect.fromLTWH(0, 0, size.width, size.height),
      paint,
    );
  }

  @override
  bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
}

3. 纹理图片传参

下面来看一下如何 Flutter 中如何将一张图片数据作为入参传递为着色器代码,比如把一张可爱女孩的照片展示到屏幕上:

着色器代码中,通过 uniform 声明 sampler2D 类型的对象表示贴图变量,通过内置的 texture 函数可以根据坐标值提取纹理的颜色;如果将其作为输出色,即可将图片原封不动地展示出来:

代码语言:javascript
复制
---->[shaders/var_03.frag]----
#version 460 core
precision mediump float;
#include <flutter/runtime_effect.glsl>

uniform vec2 uSize;
uniform sampler2D uTexture;
out vec4 fragColor;

void main() {
    vec2 coo = FlutterFragCoord().xy/uSize;
    vec4 color = texture(uTexture, coo);
    fragColor = color;
}

在 Flutter 中也是通过 setImageSampler 传入 ui.Image 对象作为贴图的数据,索引顺序从 0 开始,如果由多张图片,依次计数。

代码语言:javascript
复制
---->[lib/paint/shaders/var_demos/v3_painter.dart]----
class V3ShaderPainter extends CustomPainter {
  V3ShaderPainter({required this.shader, required this.image});

  FragmentShader shader;
  ui.Image image;


  @override
  void paint(Canvas canvas, Size size) {
    shader.setFloat(0, size.width);
    shader.setFloat(1, size.height);
    /// 传递贴图数据
    shader.setImageSampler(0, image);
    final paint = Paint()..shader = shader;
    canvas.drawRect(
      Rect.fromLTWH(0, 0, size.width, size.height),
      paint,
    );
  }

  @override
  bool shouldRepaint(covariant CustomPainter oldDelegate) => false;
}

4. 综合传参案例

最后通过一个综合小案例练习一下传参:既然 GLSL 代码中可以获得纹理图片的每个像素颜色。那么就可以通过 mix 函数 将像素颜色和另一个颜色混合

如下所示,选择颜色时进行混色;下方的进度条用于设置混色的程度,根据程度进行插值计算,视觉表现就是程度越大,混色越 "浓" 。

切换混色

调整混色程度

下面是着色器代码,定义程度 progress、尺寸 uSize、颜色 uColor、贴图 uTexture 四个变量:

代码语言:javascript
复制
---->[shaders/var_04.frag]----
#version 460 core
precision mediump float;
#include <flutter/runtime_effect.glsl>

uniform float progress;
uniform vec2 uSize;
uniform vec4 uColor;
uniform sampler2D uTexture;

out vec4 fragColor;

void main() {
    vec2 coo = FlutterFragCoord().xy / uSize;
    vec4 color = texture(uTexture, coo);
    fragColor = mix(color, uColor, progress);
}

Flutter 中通过交互更新数据,并设置对应的数据传递给着色器代码,注意参数的索引顺序要对应好:

代码语言:javascript
复制
---->[lib/paint/shaders/var_demos/v4_painter.dart]----
class V4ShaderPainter extends CustomPainter {
  V4ShaderPainter({
    required this.shader,
    required this.image,
    required this.color,
    required this.progress,
  });

  FragmentShader shader;
  ui.Image image;
  Color color;
  double progress;

  @override
  void paint(Canvas canvas, Size size) {
    shader.setFloat(0, progress);
    shader.setFloat(1, size.width);
    shader.setFloat(2, size.height);
    shader.setFloat(3, color.red / 255);
    shader.setFloat(4, color.green / 255);
    shader.setFloat(5, color.blue / 255);
    shader.setFloat(6, color.alpha / 255);

    shader.setImageSampler(0, image);
    final paint = Paint()..shader = shader;
    canvas.drawRect(
      Rect.fromLTWH(0, 0, size.width, size.height),
      paint,
    );
  }

  @override
  bool shouldRepaint(covariant CustomPainter oldDelegate) => true;
}

总得来说,Flutter 像着色器代码传递参数还是非常方便的,有了参数的加持,Flutter 就可以在交互过程中完成很多实用的功能,比如图片的特效处理,绚丽图片的生成。那本篇就到这里,感谢观看,下次再见~

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2024-02-17,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Flutter & GLSL 系列文章:
  • 1. 从尺寸入参开始说起
  • 2. 选择颜色进行渐变
  • 3. 纹理图片传参
  • 4. 综合传参案例
相关产品与服务
图数据库 KonisGraph
图数据库 KonisGraph(TencentDB for KonisGraph)是一种云端图数据库服务,基于腾讯在海量图数据上的实践经验,提供一站式海量图数据存储、管理、实时查询、计算、可视化分析能力;KonisGraph 支持属性图模型和 TinkerPop Gremlin 查询语言,能够帮助用户快速完成对图数据的建模、查询和可视化分析。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档