IT博客汇
  • 首页
  • 精华
  • 技术
  • 设计
  • 资讯
  • 扯淡
  • 权利声明
  • 登录 注册

    如何用简单的Web方法实现图片的马赛克效果

    张 鑫旭发表于 2024-05-19 16:18:16
    love 0

    by zhangxinxu from https://www.zhangxinxu.com/wordpress/?p=11187 鑫空间-鑫生活
    本文可全文转载,独立域名个人网站无需授权,但需要保留原作者、出处以及文中链接,任何网站均可摘要聚合,商用请联系授权。

    封面占位图

    一、开门见山先看实现

    在Web上实现图片的马赛克效果最简单的做法就是借助SVG滤镜。

    操作很简单,页面任意位置粘贴如下所示的SVG代码:

    <svg>
      <filter id="mosaic">
        <feFlood x="4" y="4" height="2" width="2"/>
        <feComposite width="8" height="8"/>
        <feTile result="a"/>
        <feComposite in="SourceGraphic" in2="a" operator="in"/>
        <feMorphology operator="dilate" radius="4"/>
      </filter>
    </svg>

    然后再给需要应用马赛克效果的地方使用如下所示的CSS代码即可:

    img {
      filter: url(#mosaic);
    }

    案例示意

    例如,有一张如下所示的原始图:

    在应用上述滤镜效果后:

    如果看不到效果,您可以狠狠地点击这里体验:feMorphology滤镜实现图片马赛克效果demo

    二、实现原理详解

    对SVG滤镜作用,以及这里是如何生效不感兴趣的可以跳过,直接看第三段,不使用SVG滤镜实现图片的马赛克效果。

    1. feFlood是用来添加覆盖层和透明度的,支持flood-color和flood-opacity这两个属性,如果不设置这两个属性值,那么就会以纯黑色代替。所以下面这行代码的意思是:
      <feFlood x="4" y="4" height="2" width="2"/>

      创建一个坐标(4,4),宽高都是2的黑色方形点点。

    2. feComposite是用来混合两个输入图像效果的,有点类似于遮罩相关的子属性mask-composite属性,可以决定两个输入图像重叠部分是如何显示的,交集隐藏,还是非交集隐藏,还是直接叠加这种。对于<feComposite width="8" height="8"/>这段代码,由于没有任何in/in2属性设置,因此,作用的就是<filter>元素内,自身元素之前的滤镜元素,也就是上面的feFlood。

      相同于创建了一个8×8大小的区域,然后将feFlood黑色方块放在其中。

    3. feTile可以看成是底纹元素,天然平铺。在本例中,其作用是将8×8大小的区域(含中间黑块)平铺整个原始图像,示意效果如下图所示:
      黑色点点平铺
    4. feComposite上面有讲过,这里的作用就是将黑点和原始图像混合,operator属性指的是混合方式,operator=”in”表示in输入图像和in2输入图像重叠的部分显示为in输入图像内容,不重叠的部分不显示。
      因此,截止到此滤镜的效果是这样的:

      滤镜效果分布示意

    5. 最后这个feMorphology就是上一篇文章介绍的侵蚀和扩展滤镜了,operator=”dilate”表示扩展,radius属性表示扩展半径。严格来讲,这里的扩展半径大小应该是3,也就是radius=”3″,因为8*8里面有个2*2的黑色方块,只需要扩展3像素就足够了。但是实际上的效果却和我们想要的马赛克有些不同,如下图所示,边缘融合有些生硬。

      马赛克小尺寸效果

      所以,就设置了radius=”4″,效果赞赞赞。

    是不是一下子豁然开朗了。

    三、不使用SVG滤镜实现马赛克

    小图放大的马赛克实现策略。

    在Web中,当我们给图片设置image-rendering:pixelated这个声明的时候,如果原始图尺寸比较小,设置的预览尺寸比较大,那么这个图片就会自动边缘像素化显示,所呈现的效果就非常接近马赛克效果。

    例如,我有一张只有90px宽度的头像,名为zxx_90_0824.jpg,现在,将其款苏设置为300px,同时设置图片渲染模式为pixelated,我们可以看下其实时渲染效果。

    <img src="zxx_90_0824.jpg" />
    img {
      width: 300px;
      image-rendering: pixelated;
    }

    原始图:

    原始尺寸大小

    马赛克后:

    马赛克效果

    因为图片放大比例仅3倍多,所以马赛克大小也不明显,实际操作的时候,往往5倍尺寸放大。

    目前,无论是腾讯云还是阿里云的COS存储服务,都支持参数设置图片尺寸,例如,图片url地址后面加上类似下面代码所示的尺寸查询设置:

    <img src="example.jpg?30x30" />

    不支持COS地址的图片处理

    如果我们的素材只有一张大图,则我们可以想办法把图片尺寸变小,然后自然就可以支持马赛克效果了。

    这就需要用到canvas,将原始图绘制在小尺寸画布上,然后再应用image-rendering:pixelated实现马赛克效果。

    为了方便大家学习(copy代码),我专门做了个演示页面。

    您可以狠狠地点击这里:canvas改变图片尺寸实现马赛克效果demo

    实现效果如下截图所示:

    demo页面效果示意

    image-rendering属性的兼容性

    如下截图所示,所有现代浏览器均支持。

    兼容性截图

    对于这个属性,我在《CSS新世界》这本书中,也有比较详细的介绍,有兴趣的可以前去看看。

    四、充实的一周的结语

    上篇文章有说,要弄视频,精讲下CSS世界三部曲,目前,视频号、抖音(新账号,搜“张鑫旭本人”)还有B站第一集都已经发了。

    B站地址:https://www.bilibili.com/video/BV1fw4m1D7N2/

    保持每周一更的节奏,欢迎大家点赞,关注,转发支持。

    这一周工作产出也可以,周六周日还钓了鱼,钓鱼视频(搜“最会钓鱼的程序员”)也有1万多浏览量。

    还产出了这篇博文。

    另外,还码了1万多字的小说。

    所以说,还是忙一点,找点事做人才会更舒服,舒服了,做什么事情都高效。

    每天看小说,刷短视频只能短期愉悦,停下来,什么也没留下,反而不愉悦。

    扯远了,回到文章这里,如果你觉得内容不错,欢迎点赞,欢迎转发,也欢迎,比个大大的心。

    比心

    本文为原创文章,会经常更新知识点以及修正一些错误,因此转载请保留原出处,方便溯源,避免陈旧错误知识的误导,同时有更好的阅读体验。
    本文地址:https://www.zhangxinxu.com/wordpress/?p=11187

    (本篇完)



沪ICP备19023445号-2号
友情链接