自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

使用RenderScript實(shí)現(xiàn)高斯模糊(毛玻璃/磨砂)效果

移動(dòng)開發(fā) Android
逛instagram的時(shí)候,偶然發(fā)現(xiàn),instagram的對(duì)話框設(shè)計(jì)的很有意思,它的dialog的背景竟然是毛玻璃效果的,在我看來真漂亮,恩,對(duì)話框和迪麗熱巴都漂亮??吹竭@么好的效果,當(dāng)然就要開始搞事情了,自己動(dòng)手實(shí)現(xiàn)差不多的效果。

前言

逛instagram的時(shí)候,偶然發(fā)現(xiàn),instagram的對(duì)話框設(shè)計(jì)的很有意思,如下圖:   

instagram的對(duì)話框 

它的dialog的背景竟然是毛玻璃效果的,在我看來真漂亮,恩,對(duì)話框和迪麗熱巴都漂亮😂??吹竭@么好的效果,當(dāng)然就要開始搞事情了,自己動(dòng)手實(shí)現(xiàn)差不多的效果。最終的實(shí)現(xiàn)效果如下圖: 

 

 

  

 

 

 

分別實(shí)現(xiàn)了對(duì)話框背景的虛化和手動(dòng)調(diào)節(jié)虛化程度。

實(shí)現(xiàn)方法對(duì)比

最開始想要實(shí)現(xiàn)毛玻璃效果時(shí),我是一臉懵逼的,不知道如何下手。幸虧,有***的Google。搜索之后發(fā)現(xiàn)常見的實(shí)現(xiàn)方法有4種,分別是:

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

處理一整張圖片這么大計(jì)算量的工作,openGL的性能***,而用java實(shí)現(xiàn)肯定是最差的了。而RenderScript和NDK的性能相當(dāng),但是你懂得,NDK和openGL我無可奈何,綜合考慮,RenderScript應(yīng)該是最適合的。

但并不是說RenderScript就是完全沒有問題的:

  1. 模糊半徑(radius)越大,性能要求越高,模糊半徑不能超過25,所以并不能得到模糊度非常高的圖片。
  2. ScriptIntrinsicBlur在API 17時(shí)才被引入,如果需要在Android 4.2以下的設(shè)備上實(shí)現(xiàn),就需要引入RenderScript Support Library,當(dāng)然,安裝包體積會(huì)相應(yīng)的增大。

RenderScript實(shí)現(xiàn)

首先在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提供了一些可以幫助我們快速實(shí)現(xiàn)各種圖片處理的操作類,例如,ScriptIntrinsicBlur,可以簡單高效實(shí)現(xiàn) 高斯模糊效果。

  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,愉快地根據(jù)SeekBar的值,實(shí)現(xiàn)不同程度的模糊了。

  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的實(shí)現(xiàn)有一點(diǎn)不同的是,我沒有截取整個(gè)頁面的bitmap,只是截取了actionbar下的內(nèi)容,如果一定要實(shí)現(xiàn)一樣的效果,調(diào)整一下頁面的布局就可以了。這里不多說了。

是不是很簡單呢?

輪子

除了RenderScript外,還有一些優(yōu)秀的輪子:

BlurTestAndroid對(duì)不同類庫的實(shí)現(xiàn)方式、采取的算法和所耗費(fèi)的時(shí)間做了統(tǒng)計(jì)和比較,你也可以下載它的demo app,自行測(cè)試。 

 

 

 

示例代碼在這里:GaussianBlur 

責(zé)任編輯:龐桂玉 來源: segmentfault
相關(guān)推薦

2017-02-14 17:29:42

Android毛玻璃虛化效果

2010-01-21 15:48:38

C++語言

2021-07-06 06:07:14

CSS 技巧背景

2022-01-07 07:35:28

CSS 技巧磨砂玻璃

2016-08-30 21:36:56

JavascriptCSSWeb

2024-10-10 16:02:34

2022-03-01 17:24:33

iOS兼容性方案

2017-04-13 10:03:29

Java高斯模糊圖像

2017-01-09 09:47:40

Windows10AeroUI

2013-10-28 16:37:38

Windows 8.1技巧

2014-04-02 10:29:12

iOS 7模糊效果

2023-05-26 07:08:05

CSS模糊實(shí)現(xiàn)文字

2012-11-15 09:43:08

開發(fā)算法高斯模糊

2018-05-08 23:46:15

微軟語言Windows

2019-06-24 09:10:31

Windows操作系統(tǒng)Windows 10

2021-08-30 06:20:39

CSS 技巧3D 效果

2025-01-10 08:38:16

2022-01-09 17:05:54

Windows 11操作系統(tǒng)微軟

2013-10-24 10:10:21

Android支持庫

2023-09-26 21:53:27

Java圖像處理
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)