Process image in background thread


Last article "Sharpen image, using Convolution Matrix" perform image processing in main thread (or called UI Thread), system will not respond any other user action before the task finished - It not a good practice! If your activity take long time not responding, it will be killed by the system.



It will be modified to be performed in background Runnable. Before the image process finished, a ProgressBar will be shown to indicate that it's running.



AndroidImageProcessingActivity.java
package com.AndroidImageProcessing;

import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Color;
import android.os.Bundle;
import android.os.Handler;
import android.view.View;
import android.widget.ImageView;
import android.widget.ProgressBar;

public class AndroidImageProcessingActivity extends Activity {

final static int KERNAL_WIDTH = 3;
final static int KERNAL_HEIGHT = 3;

int[][] kernalBlur ={
{0, -1, 0},
{-1, 5, -1},
{0, -1, 0}
};

ImageView imageSource, imageAfter;
Bitmap bitmap_Source;
ProgressBar progressBar;

private Handler handler;
Bitmap afterSharpen;

/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
imageSource = (ImageView)findViewById(R.id.imageSource);
imageAfter = (ImageView)findViewById(R.id.imageAfter);
progressBar = (ProgressBar)findViewById(R.id.progressBar);

bitmap_Source = BitmapFactory.decodeResource(getResources(), R.drawable.testpicture);

handler = new Handler();
StratBackgroundProcess();
}

private void StratBackgroundProcess(){

Runnable runnable = new Runnable(){

@Override
public void run() {
afterSharpen = processingBitmap(bitmap_Source, kernalBlur);

handler.post(new Runnable(){

@Override
public void run() {
progressBar.setVisibility(View.GONE);
imageAfter.setImageBitmap(afterSharpen);
}

});
}
};
new Thread(runnable).start();
}

private Bitmap processingBitmap(Bitmap src, int[][] knl){
Bitmap dest = Bitmap.createBitmap(
src.getWidth(), src.getHeight(), src.getConfig());

int bmWidth = src.getWidth();
int bmHeight = src.getHeight();
int bmWidth_MINUS_2 = bmWidth - 2;
int bmHeight_MINUS_2 = bmHeight - 2;

for(int i = 1; i <= bmWidth_MINUS_2; i++){
for(int j = 1; j <= bmHeight_MINUS_2; j++){

//get the surround 3*3 pixel of current src[i][j] into a matrix subSrc[][]
int[][] subSrc = new int[KERNAL_WIDTH][KERNAL_HEIGHT];
for(int k = 0; k < KERNAL_WIDTH; k++){
for(int l = 0; l < KERNAL_HEIGHT; l++){
subSrc[k][l] = src.getPixel(i-1+k, j-1+l);
}
}

//subSum = subSrc[][] * knl[][]
int subSumA = 0;
int subSumR = 0;
int subSumG = 0;
int subSumB = 0;

for(int k = 0; k < KERNAL_WIDTH; k++){
for(int l = 0; l < KERNAL_HEIGHT; l++){
subSumA += Color.alpha(subSrc[k][l]) * knl[k][l];
subSumR += Color.red(subSrc[k][l]) * knl[k][l];
subSumG += Color.green(subSrc[k][l]) * knl[k][l];
subSumB += Color.blue(subSrc[k][l]) * knl[k][l];
}
}

if(subSumA<0){
subSumA = 0;
}else if(subSumA>255){
subSumA = 255;
}

if(subSumR<0){
subSumR = 0;
}else if(subSumR>255){
subSumR = 255;
}

if(subSumG<0){
subSumG = 0;
}else if(subSumG>255){
subSumG = 255;
}

if(subSumB<0){
subSumB = 0;
}else if(subSumB>255){
subSumB = 255;
}

dest.setPixel(i, j, Color.argb(
subSumA,
subSumR,
subSumG,
subSumB));
}
}

return dest;
}

}

Modify main.xml to add a ProgressBar overlap imageAfter in a FrameLayout.
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >

<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/hello" />
<ScrollView
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<LinearLayout
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="Original" />
<ImageView
android:id="@+id/imageSource"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/testpicture"/>
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="Result" />
<FrameLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content">
<ImageView
android:id="@+id/imageAfter"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
<ProgressBar
android:id="@+id/progressBar"
style="?android:attr/progressBarStyleLarge"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
</FrameLayout>

</LinearLayout>
</ScrollView>

</LinearLayout>


0 Response to "Process image in background thread"

Posting Komentar