使用RenderScript实现高斯模糊(毛玻璃/磨砂)效果

前言

逛instagram的时候,偶然发现,instagram的对话框设计的很有意思,如下图:  

它的dialog的背景竟然是毛玻璃效果的,在我看来真漂亮,恩,对话框和迪丽热巴都漂亮。看到这么好的效果,当然就要开始搞事情了,自己动手实现差不多的效果。最终的实现效果如下图:

 

分别实现了对话框背景的虚化和手动调节虚化程度。

实现方法对比

最开始想要实现毛玻璃效果时,我是一脸懵逼的,不知道如何下手。幸亏,有万能的Google。搜索之后发现常见的实现方法有4种,分别是:

  • RenderScript
  • Java算法
  • NDK算法
  • openGL

处理一整张图片这么大计算量的工作,openGL的性能最好,而用java实现肯定是最差的了。而RenderScript和NDK的性能相当,但是你懂得,NDK和openGL我无可奈何,综合考虑,RenderScript应该是最适合的。

但并不是说RenderScript就是完全没有问题的:

  1. 模糊半径(radius)越大,性能要求越高,模糊半径不能超过25,所以并不能得到模糊度非常高的图片。
  2. ScriptIntrinsicBlur在API 17时才被引入,如果需要在Android 4.2以下的设备上实现,就需要引入RenderScript Support Library,当然,安装包体积会相应的增大。

RenderScript实现

首先在app目录下build.gradle文件中添加如下代码:


  1. defaultConfig { 
  2.         applicationId "io.github.marktony.gaussianblur" 
  3.         minSdkVersion 19 
  4.         targetSdkVersion 25 
  5.         versionCode 1 
  6.         versionName "1.0" 
  7.         renderscriptTargetApi 19 
  8.         renderscriptSupportModeEnabled true 
  9.     }  

RenderScriptIntrinsics提供了一些可以帮助我们快速实现各种图片处理的操作类,例如,ScriptIntrinsicBlur,可以简单高效实现 高斯模糊效果。


  1. package io.github.marktony.gaussianblur; 
  2.  
  3. import android.content.Context; 
  4. import android.graphics.Bitmap; 
  5. import android.support.annotation.IntRange; 
  6. import android.support.annotation.NonNull; 
  7. import android.support.v8.renderscript.Allocation; 
  8. import android.support.v8.renderscript.Element; 
  9. import android.support.v8.renderscript.RenderScript; 
  10. import android.support.v8.renderscript.ScriptIntrinsicBlur; 
  11.  
  12. public class RenderScriptGaussianBlur { 
  13.  
  14.     private RenderScript renderScript; 
  15.  
  16.     public RenderScriptGaussianBlur(@NonNull Context context) { 
  17.         this.renderScript = RenderScript.create(context); 
  18.     } 
  19.  
  20.     public Bitmap gaussianBlur(@IntRange(from = 1, to = 25) int radius, Bitmap original) { 
  21.         Allocation input = Allocation.createFromBitmap(renderScript, original); 
  22.         Allocation output = Allocation.createTyped(renderScript, input.getType()); 
  23.         ScriptIntrinsicBlur scriptIntrinsicBlur = ScriptIntrinsicBlur.create(renderScript, Element.U8_4(renderScript)); 
  24.         scriptIntrinsicBlur.setRadius(radius); 
  25.         scriptIntrinsicBlur.setInput(input); 
  26.         scriptIntrinsicBlur.forEach(output); 
  27.         output.copyTo(original); 
  28.         return original; 
  29.     } 
  30.  
  31. }  

然后就可以直接使用RenderScriptGaussianBlur,愉快地根据SeekBar的值,实现不同程度的模糊了。


  1. package io.github.marktony.gaussianblur; 
  2.  
  3. import android.content.DialogInterface; 
  4. import android.graphics.Bitmap; 
  5. import android.graphics.BitmapFactory; 
  6. import android.support.v7.app.AlertDialog; 
  7. import android.support.v7.app.AppCompatActivity; 
  8. import android.os.Bundle; 
  9. import android.util.Log; 
  10. import android.view.View; 
  11. import android.view.Window; 
  12. import android.view.WindowManager; 
  13. import android.widget.FrameLayout; 
  14. import android.widget.ImageView; 
  15. import android.widget.LinearLayout; 
  16. import android.widget.SeekBar; 
  17. import android.widget.TextView; 
  18.  
  19. public class MainActivity extends AppCompatActivity { 
  20.  
  21.     private ImageView imageView; 
  22.     private ImageView container; 
  23.     private LinearLayout layout; 
  24.     private TextView textViewProgress; 
  25.     private RenderScriptGaussianBlur blur; 
  26.  
  27.     @Override 
  28.     protected void onCreate(Bundle savedInstanceState) { 
  29.         super.onCreate(savedInstanceState); 
  30.         setContentView(R.layout.activity_main); 
  31.  
  32.         imageView = (ImageView) findViewById(R.id.imageView); 
  33.         container = (ImageView) findViewById(R.id.container); 
  34.  
  35.         container.setVisibility(View.GONE); 
  36.  
  37.         layout = (LinearLayout) findViewById(R.id.layout); 
  38.  
  39.         layout.setVisibility(View.VISIBLE); 
  40.  
  41.         SeekBar seekBar = (SeekBar) findViewById(R.id.seekBar); 
  42.         textViewProgress = (TextView) findViewById(R.id.textViewProgress); 
  43.         TextView textViewDialog = (TextView) findViewById(R.id.textViewDialog); 
  44.         blur = new RenderScriptGaussianBlur(MainActivity.this); 
  45.  
  46.         seekBar.setMax(25); 
  47.         seekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() { 
  48.             @Override 
  49.             public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { 
  50.                 textViewProgress.setText(String.valueOf(progress)); 
  51.             } 
  52.  
  53.             @Override 
  54.             public void onStartTrackingTouch(SeekBar seekBar) { 
  55.  
  56.             } 
  57.  
  58.             @Override 
  59.             public void onStopTrackingTouch(SeekBar seekBar) { 
  60.                 int radius = seekBar.getProgress(); 
  61.                 if (radius < 1) { 
  62.                     radius = 1; 
  63.                 } 
  64.                 Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image); 
  65.                 imageView.setImageBitmap(blur.gaussianBlur(radius, bitmap)); 
  66.             } 
  67.         }); 
  68.  
  69.         textViewDialog.setOnClickListener(new View.OnClickListener() { 
  70.             @Override 
  71.             public void onClick(View v) { 
  72.  
  73.                 container.setVisibility(View.VISIBLE); 
  74.  
  75.                 layout.setDrawingCacheEnabled(true); 
  76.                 layout.setDrawingCacheQuality(View.DRAWING_CACHE_QUALITY_LOW); 
  77.  
  78.                 Bitmap bitmap = layout.getDrawingCache(); 
  79.  
  80.                 container.setImageBitmap(blur.gaussianBlur(25, bitmap)); 
  81.  
  82.                 layout.setVisibility(View.INVISIBLE); 
  83.  
  84.                 AlertDialog dialog = new AlertDialog.Builder(MainActivity.this).create(); 
  85.                 dialog.setTitle("Title"); 
  86.                 dialog.setMessage("Message"); 
  87.                 dialog.setButton(DialogInterface.BUTTON_POSITIVE, "OK", new DialogInterface.OnClickListener() { 
  88.                     @Override 
  89.                     public void onClick(DialogInterface dialog, int which) { 
  90.                         dialog.dismiss(); 
  91.                     } 
  92.                 }); 
  93.                 dialog.setButton(DialogInterface.BUTTON_NEGATIVE, "Cancel", new DialogInterface.OnClickListener() { 
  94.                     @Override 
  95.                     public void onClick(DialogInterface dialog, int which) { 
  96.  
  97.                     } 
  98.                 }); 
  99.                 dialog.setOnCancelListener(new DialogInterface.OnCancelListener() { 
  100.                     @Override 
  101.                     public void onCancel(DialogInterface dialog) { 
  102.  
  103.                     } 
  104.                 }); 
  105.  
  106.                 dialog.setOnCancelListener(new DialogInterface.OnCancelListener() { 
  107.                     @Override 
  108.                     public void onCancel(DialogInterface dialog) { 
  109.                         container.setVisibility(View.GONE); 
  110.                         layout.setVisibility(View.VISIBLE); 
  111.                     } 
  112.                 }); 
  113.  
  114.                 dialog.show(); 
  115.             } 
  116.         }); 
  117.  
  118.     } 
  119. }  

在代码里做了一些view的可见性的操作,比较简单,相信你能看懂的。和instagram中dialog的实现有一点不同的是,我没有截取整个页面的bitmap,只是截取了actionbar下的内容,如果一定要实现一样的效果,调整一下页面的布局就可以了。这里不多说了。

是不是很简单呢?

轮子

除了RenderScript外,还有一些优秀的轮子:

BlurTestAndroid对不同类库的实现方式、采取的算法和所耗费的时间做了统计和比较,你也可以下载它的demo app,自行测试。

作者:Tonny

来源:51CTO

时间: 2024-11-02 10:04:33

使用RenderScript实现高斯模糊(毛玻璃/磨砂)效果的相关文章

Android 高仿微信语音聊天页面高斯模糊(毛玻璃效果)_Android

目前的应用市场上,使用毛玻璃效果的APP随处可见,比如用过微信语音聊天的人可以发现,语音聊天页面就使用了高斯模糊效果. 先看下效果图:   仔细观察上图,我们可以发现,背景图以用户头像为模板,对其进行了高斯模糊,并把它作为整个页面的背景色. 关于Android如何快速实现高斯模糊(毛玻璃效果),网上一堆相关介绍,可参考下面文章一种快速毛玻璃虚化效果实现–Android. 下面直接给出模糊化工具类(已验证可行): import android.graphics.Bitmap; /** * 快速模糊

Android RenderScript实现高斯模糊_Android

昨天看了下RenderScript的官方文档,发现RenderScript这厮有点牛逼.无意中发现ScriptIntrinsic这个抽象类,有些很有用的子类.其中有个子类叫ScriptIntrinsicBlur类,大致就是将图片实现高斯模糊. ScriptIntrinsic的申明: ScriptIntrinsicBlur类的申明: 加上结合着看了下SDK中的samples,自己写了个高斯模糊. ( sample的具体位置为: SDK目录/samples/android-19/renderscri

Photoshop给照片添加富有艺术感的磨砂效果

  Photoshop给照片添加富有艺术感的磨砂效果          自Photoshop CS开始,他们就提供了"匹配颜色"功能,可以将一张图片的色盘,调进另一张图片里,可以用作统一色调.纠正色彩甚至使日间风景变成黄昏日落等,亦能够为你的照片增添独特的色调,尤其是如果你喜欢那些有趣的用色,诸如名画家的作品,既大胆又抢眼,就可以来试试了. 首先,在Photoshop上分别打开下面两张图片,我会用下图的偏紫用色来调节上图. 然后,回到想调节的照片,再打开图像>调整>匹配颜色

Photoshop设计古典磨砂效果图文教程

主要使用Photoshop制作秋心雨图文实例教程,整体的设计风格以复古磨砂效果为主,教程的素材在压缩包中,喜欢的朋友让我们一起来学习吧. 分类: PS入门教程

develop-iOS第三方APP能不能做出这种桌面背景实时毛玻璃的效果?

问题描述 iOS第三方APP能不能做出这种桌面背景实时毛玻璃的效果? 解决方案 主要是性能问题,如果你的背景不是每一秒都在变化,而只是某些时候才发生变化的话,可以在特定时候更新毛玻璃背景就好了 静态毛玻璃的例子网上有很多,如果你希望实时更新,那就比较困难了,第三方app基本都是静态毛玻璃,只是看上去像是动态的 如果用系统的api也有系统版本和设备的限制.除非你不考虑兼容问题 解决方案二: iOS8以上支持的较好.有些效果要在真机才能用.

Android 高仿微信语音聊天页面高斯模糊(毛玻璃效果)

目前的应用市场上,使用毛玻璃效果的APP随处可见,比如用过微信语音聊天的人可以发现,语音聊天页面就使用了高斯模糊效果. 先看下效果图: 仔细观察上图,我们可以发现,背景图以用户头像为模板,对其进行了高斯模糊,并把它作为整个页面的背景色. 关于Android如何快速实现高斯模糊(毛玻璃效果),网上一堆相关介绍,可参考下面文章一种快速毛玻璃虚化效果实现–Android. 下面直接给出模糊化工具类(已验证可行): import android.graphics.Bitmap; /** * 快速模糊化工

Fireworks表现Vista毛玻璃朦胧效果

  今天我们用Fireworks制作类似vista毛玻璃半透明的效果,局部效果图如下: 1.建立画布,导入背景素材 2.绘制圆角矩形,垂直线性渐变填充 3.绘制矩形高光部分,复制矩形,原位粘贴,并删除下半部分,进行垂直线性渐变,调整透明度 4.开始制作毛玻璃效果,线性渐变绘制一个圆形,羽化10px 5.添加杂点8px 6.选择先前绘制矩形,在刚才画的圆形上点击反键-编辑-粘贴为蒙版 7.得到如下效果 8.同上步骤再绘制一个矩形.放射性填充.羽化及添加杂点步骤 9.在msn小人这块区域绘制一个椭圆

winform下实现win7 Aero磨砂效果

问题描述 usingSystem;usingSystem.Collections.Generic;usingSystem.ComponentModel;usingSystem.Data;usingSystem.Drawing;usingSystem.Text;usingSystem.Windows.Forms;usingSystem.ServiceModel;usingSystem.Runtime.InteropServices;namespaceMyWeather{publicpartialc

Android 实现图片模糊、高斯模糊、毛玻璃效果的三种方法_Android

在前几天写过一个使用glide-transformations的方法实现高斯模糊的方法,今天偶然间有发现一个大神写的另一个方法,感觉挺不错的,分享一下: 效果图: 原文链接:点击访问 这使用也很简单,导入依赖,使用模糊方法就行,就这两步搞定 依赖: compile 'net.qiujuer.genius:blur:2.0.0-beta4' 实现方法有三种,第一个是Java实现的,第二个和第三个是调用C语言实现的 ,具体的区别也就是代码执行的耗时操作时间,从图片中可以看出Java使用时间远大于c运