欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

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

程序员文章站 2024-03-07 15:11:21
目前的应用市场上,使用毛玻璃效果的app随处可见,比如用过微信语音聊天的人可以发现,语音聊天页面就使用了高斯模糊效果。 先看下效果图:   仔细观察上图,我...

目前的应用市场上,使用毛玻璃效果的app随处可见,比如用过微信语音聊天的人可以发现,语音聊天页面就使用了高斯模糊效果。

先看下效果图:

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

仔细观察上图,我们可以发现,背景图以用户头像为模板,对其进行了高斯模糊,并把它作为整个页面的背景色。

关于android如何快速实现高斯模糊(毛玻璃效果),网上一堆相关介绍,可参考下面文章一种快速毛玻璃虚化效果实现–android。 下面直接给出模糊化工具类(已验证可行):

import android.graphics.bitmap;
/**
* 快速模糊化工具
*/
public class fastblur {
public static bitmap doblur(bitmap sentbitmap, int radius, boolean canreuseinbitmap) {
bitmap bitmap;
if (canreuseinbitmap) {
bitmap = sentbitmap;
} else {
bitmap = sentbitmap.copy(sentbitmap.getconfig(), true);
}
if (radius < 1) {
return (null);
}
int w = bitmap.getwidth();
int h = bitmap.getheight();
int[] pix = new int[w * h];
bitmap.getpixels(pix, 0, w, 0, 0, w, h);
int wm = w - 1;
int hm = h - 1;
int wh = w * h;
int div = radius + radius + 1;
int r[] = new int[wh];
int g[] = new int[wh];
int b[] = new int[wh];
int rsum, gsum, bsum, x, y, i, p, yp, yi, yw;
int vmin[] = new int[math.max(w, h)];
int divsum = (div + 1) >> 1;
divsum *= divsum;
int dv[] = new int[256 * divsum];
for (i = 0; i < 256 * divsum; i++) {
dv[i] = (i / divsum);
}
yw = yi = 0;
int[][] stack = new int[div][3];
int stackpointer;
int stackstart;
int[] sir;
int rbs;
int r1 = radius + 1;
int routsum, goutsum, boutsum;
int rinsum, ginsum, binsum;
for (y = 0; y < h; y++) {
rinsum = ginsum = binsum = routsum = goutsum = boutsum = rsum = gsum = bsum = 0;
for (i = -radius; i <= radius; i++) {
p = pix[yi + math.min(wm, math.max(i, 0))];
sir = stack[i + radius];
sir[0] = (p & 0xff0000) >> 16;
sir[1] = (p & 0x00ff00) >> 8;
sir[2] = (p & 0x0000ff);
rbs = r1 - math.abs(i);
rsum += sir[0] * rbs;
gsum += sir[1] * rbs;
bsum += sir[2] * rbs;
if (i > 0) {
rinsum += sir[0];
ginsum += sir[1];
binsum += sir[2];
} else {
routsum += sir[0];
goutsum += sir[1];
boutsum += sir[2];
}
}
stackpointer = radius;
for (x = 0; x < w; x++) {
r[yi] = dv[rsum];
g[yi] = dv[gsum];
b[yi] = dv[bsum];
rsum -= routsum;
gsum -= goutsum;
bsum -= boutsum;
stackstart = stackpointer - radius + div;
sir = stack[stackstart % div];
routsum -= sir[0];
goutsum -= sir[1];
boutsum -= sir[2];
if (y == 0) {
vmin[x] = math.min(x + radius + 1, wm);
}
p = pix[yw + vmin[x]];
sir[0] = (p & 0xff0000) >> 16;
sir[1] = (p & 0x00ff00) >> 8;
sir[2] = (p & 0x0000ff);
rinsum += sir[0];
ginsum += sir[1];
binsum += sir[2];
rsum += rinsum;
gsum += ginsum;
bsum += binsum;
stackpointer = (stackpointer + 1) % div;
sir = stack[(stackpointer) % div];
routsum += sir[0];
goutsum += sir[1];
boutsum += sir[2];
rinsum -= sir[0];
ginsum -= sir[1];
binsum -= sir[2];
yi++;
}
yw += w;
}
for (x = 0; x < w; x++) {
rinsum = ginsum = binsum = routsum = goutsum = boutsum = rsum = gsum = bsum = 0;
yp = -radius * w;
for (i = -radius; i <= radius; i++) {
yi = math.max(0, yp) + x;
sir = stack[i + radius];
sir[0] = r[yi];
sir[1] = g[yi];
sir[2] = b[yi];
rbs = r1 - math.abs(i);
rsum += r[yi] * rbs;
gsum += g[yi] * rbs;
bsum += b[yi] * rbs;
if (i > 0) {
rinsum += sir[0];
ginsum += sir[1];
binsum += sir[2];
} else {
routsum += sir[0];
goutsum += sir[1];
boutsum += sir[2];
}
if (i < hm) {
yp += w;
}
}
yi = x;
stackpointer = radius;
for (y = 0; y < h; y++) {
// preserve alpha channel: ( 0xff000000 & pix[yi] )
pix[yi] = (0xff000000 & pix[yi]) | (dv[rsum] << 16) | (dv[gsum] << 8) | dv[bsum];
rsum -= routsum;
gsum -= goutsum;
bsum -= boutsum;
stackstart = stackpointer - radius + div;
sir = stack[stackstart % div];
routsum -= sir[0];
goutsum -= sir[1];
boutsum -= sir[2];
if (x == 0) {
vmin[y] = math.min(y + r1, hm) * w;
}
p = x + vmin[y];
sir[0] = r[p];
sir[1] = g[p];
sir[2] = b[p];
rinsum += sir[0];
ginsum += sir[1];
binsum += sir[2];
rsum += rinsum;
gsum += ginsum;
bsum += binsum;
stackpointer = (stackpointer + 1) % div;
sir = stack[stackpointer];
routsum += sir[0];
goutsum += sir[1];
boutsum += sir[2];
rinsum -= sir[0];
ginsum -= sir[1];
binsum -= sir[2];
yi += w;
}
}
bitmap.setpixels(pix, 0, w, 0, 0, w, h);
return (bitmap);
}
}

那么,我们使用这个工具类,就可以高仿微信聊天页面了么? 答案是no。

我们先看下直接使用该工具类能实现的效果:

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

我们可以看出来,通过该工具类,能实现图片的毛玻璃效果,可是并不理想,因为微信的背景颜色偏暗,而我们模糊化得到的图片颜色偏淡。 效果有些不尽如人意。 了解ios的人,或许知道,ios实现高斯模糊有直接的api: uiblureffectstyleextralight、uiblureffectstylelight 、uiblureffectstyledark,而uiblureffectstyledark就可以直接实现比较暗的毛玻璃效果。 那放在android这边,该如何办呢?

我这里采用的是“曲线求国”的策略,要想背景色偏暗,我们使用framelayout布局,在其中添加一个比较暗的一层view,就可以实现啦。

<?xml version="1.0" encoding="utf-8"?>
<framelayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitssystemwindows="true"
android:orientation="vertical">
<imageview
android:id="@+id/imgbg"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:adjustviewbounds="true"
android:maxheight="1500dp"
android:maxwidth="1000dp"
android:scaletype="centercrop" />
<imageview
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#90000000" />
<relativelayout
android:id="@+id/rootview"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitssystemwindows="true"
android:orientation="vertical">
<imageview
android:id="@+id/imguserhead"
android:layout_width="100dp"
android:layout_height="100dp"
android:layout_centerhorizontal="true"
android:layout_margintop="80dp"
android:scaletype="fitxy"
android:src="@mipmap/ic_launcher" />
<textview
android:id="@+id/textusername"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@id/imguserhead"
android:layout_centerhorizontal="true"
android:layout_margintop="30dp"
android:gravity="center_horizontal"
android:text="静音"
android:textcolor="#ffffff"
android:textsize="24dp" />
</relativelayout>
</framelayout>

上面代码中,可以看到,我们添加了这么一个imageview:

<imageview
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#90000000" />

最重实现的效果图如下:

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

源码地址:https://github.com/zuiwuyuan/fastblur_voicechat

以上所述是小编给大家介绍的android 高仿微信语音聊天页面高斯模糊效果,希望对大家有所帮助