Skip to main content

Image Compression like WhatsApp in android

How to compress the image without loosing the quality.
You can compress the image just like whatsapp like quality.

File externalFile = new File(Environment.getExternalStorageDirectory(), 
                                                       "ProfilePic/helloimg.jpg");
Uri externaluri = Uri.parse(externalFile.getPath());
compressImage(externaluri.toString(), "helloimg");

// Method to compress Image Compress image
public String compressImage(String imageUri, String imagenameName) {

    String filePath = getRealPathFromURI(imageUri);
    String fileName = imagenameName;
    Bitmap scaledBitmap = null;
    BitmapFactory.Options options = new BitmapFactory.Options();
    //by setting this field as true, the actual bitmap pixels are not loaded in the memory. 
      Just the bounds are loaded. If
    //you try the use the bitmap here, you will get null.
    options.inJustDecodeBounds = true;
    Bitmap bmp = BitmapFactory.decodeFile(filePath, options);
    int actualHeight = options.outHeight;
    int actualWidth = options.outWidth;
    //max Height and width values of the compressed image is taken as 816x612
    float maxHeight = 816.0f;    
    float maxWidth = 612.0f;    
    float imgRatio = actualWidth / actualHeight;    
    float maxRatio = maxWidth / maxHeight;
    //width and height values are set maintaining the aspect ratio of the image
    if (actualHeight > maxHeight || actualWidth > maxWidth) {
        if (imgRatio < maxRatio) {
            imgRatio = maxHeight / actualHeight;            
            actualWidth = (int) (imgRatio * actualWidth);            
            actualHeight = (int) maxHeight;        
        } else if (imgRatio > maxRatio) {
            imgRatio = maxWidth / actualWidth;            
            actualHeight = (int) (imgRatio * actualHeight);            
            actualWidth = (int) maxWidth;        
        } else {
            actualHeight = (int) maxHeight;            
            actualWidth = (int) maxWidth;
        }
    }

    //setting inSampleSize value allows to load a scaled down version of the original image    
    options.inSampleSize = calculateInSampleSize(options, actualWidth, actualHeight);

    //inJustDecodeBounds set to false to load the actual bitmap    
    options.inJustDecodeBounds = false;
    //this options allow android to claim the bitmap memory if it runs low on memory    
    options.inPurgeable = true;    
    options.inInputShareable = true;    
    options.inTempStorage = new byte[16 * 1024];
    try {
        //load the bitmap from its path        
        bmp = BitmapFactory.decodeFile(filePath, options);    
    } catch (OutOfMemoryError exception) {
        exception.printStackTrace();
    }
    try {
        scaledBitmap = Bitmap.createBitmap(actualWidth, actualHeight, 
                                                            Bitmap.Config.ARGB_8888);    
    } catch (OutOfMemoryError exception) {
        exception.printStackTrace();    
    }

    float ratioX = actualWidth / (float) options.outWidth;    
    float ratioY = actualHeight / (float) options.outHeight;    
    float middleX = actualWidth / 2.0f;    
    float middleY = actualHeight / 2.0f;
    Matrix scaleMatrix = new Matrix();    
    scaleMatrix.setScale(ratioX, ratioY, middleX, middleY);
    Canvas canvas = new Canvas(scaledBitmap);    
    canvas.setMatrix(scaleMatrix);    
    canvas.drawBitmap(bmp, middleX - bmp.getWidth() / 2, middleY - bmp.getHeight() / 2, 
                                                       new Paint(Paint.FILTER_BITMAP_FLAG));
    //check the rotation of the image and display it properly    
    ExifInterface exif;    
    try {
        exif = new ExifInterface(filePath);
        int orientation = exif.getAttributeInt(
                ExifInterface.TAG_ORIENTATION, 0);        
            Matrix matrix = new Matrix();        
            if (orientation == 6) {
                 matrix.postRotate(90);        
            } else if (orientation == 3) {
                 matrix.postRotate(180);        
            } else if (orientation == 8) {
                 matrix.postRotate(270);        
            }
        scaledBitmap = Bitmap.createBitmap(scaledBitmap, 0, 0,scaledBitmap.getWidth(), 
                                     scaledBitmap.getHeight(), matrix, true);    
    } catch (IOException e) {
        e.printStackTrace();    
    }

    FileOutputStream out = null;
    File file = new File(Environment.getExternalStorageDirectory().getPath(), 
                                                            "ProfilePic");    
    if (!file.exists()) {
        file.mkdirs();    
    }
    String uriSting = (file.getAbsolutePath() + "/" + fileName + ".jpg");
    String filename = uriSting;
    try {
        out = new FileOutputStream(filename);
        //write the compressed bitmap at the destination specified by filename.        
        scaledBitmap.compress(Bitmap.CompressFormat.JPEG, 50, out);
    } catch (FileNotFoundException e) {
        e.printStackTrace();    
    }

    File tempFile = new File("" + imagePath);    imageSize = tempFile.length() / 1024;    
    return filename;
}

private String getRealPathFromURI(String contentURI) {
    Uri contentUri = Uri.parse(contentURI);    
    Cursor cursor = getContentResolver().query(contentUri, null, null, null, null);    
    if (cursor == null) {
        return contentUri.getPath();    
    } else {
        cursor.moveToFirst();        
        int index = cursor.getColumnIndex(MediaStore.Images.ImageColumns.DATA);        
        return cursor.getString(index);    
    }
}

public int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
    final int height = options.outHeight;    
    final int width = options.outWidth;    
    int inSampleSize = 1;
    if (height > reqHeight || width > reqWidth) {
        final int heightRatio = Math.round((float) height / (float) reqHeight);        
        final int widthRatio = Math.round((float) width / (float) reqWidth);        
        inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;    
    }
    final float totalPixels = width * height;    
    final float totalReqPixelsCap = reqWidth * reqHeight * 2;    
    while (totalPixels / (inSampleSize * inSampleSize) > totalReqPixelsCap) {
        inSampleSize++;    
    }

    return inSampleSize;
}




Comments

  1. Sorry I'd like to ask with your tutorial on github about compression image.
    What method do you use in that compression and what algorithm do you use?
    I am interested to learn it.
    Thank you ;)

    ReplyDelete
  2. Hello,
    I have created my own online image compressor.
    can you please give me suggestions to improve it.

    Image Compressor

    ReplyDelete

Post a Comment

Popular posts from this blog

Vertical AutoScrolling TextView in Android

In android by default we can scroll the text in horizontal using marquee in layout, but if we want to scroll the text in vertical its not possible by default. So here we will learn to create a custom TextView which will auto-scroll in vertical direction. Source Code:  VerticalScrollingTextView-Android Create a AutoScrollingTextView.class which extends TextView: @SuppressLint ( "AppCompatCustomView" ) public class AutoScrollingTextView extends TextView { private static final float DEFAULT_SPEED = 65.0f ; public Scroller scroller ; public float speed = DEFAULT_SPEED ; public boolean continuousScrolling = true; public AutoScrollingTextView (Context context) { super (context) ; init( null, 0 ) ; scrollerInstance(context) ; } public AutoScrollingTextView (Context context , AttributeSet attrs) { super (context , attrs) ; init(attrs , 0 ) ; scr...

Flexbox inside the RecyclerView as a LayoutManager (FlexboxLayoutManager).

Currently google has release the Flexbox which can be used for building flexible layouts using FlexboxLayout, it can be interpreted as an advanced LinearLayout because both layouts align their child views sequentially. For more detail on this flexbox-layout But here we are gonna work on Flexbox with RecyclerView. Flexbox with a large number of items in a scrollable container! Let's first see what are the Supported attributes / features comparison Due to some characteristics of the RecyclerView, some Flexbox attributes are not available/not implemented to the FlexboxLayoutManager. Here is a quick overview of the attributes/features comparison between the two containers. Attribute / Feature FlexboxLayout                FlexboxLayoutManager (RecyclerView) flexDirection flexWrap (except wrap_reverse ) justifyContent alignItems alignContent - layout_order - layout_fle...

Android RecyclerView and StaggeredGridLayoutManager with Picasso/Glide

This project is there in GitHub https://github.com/yuvaraj119/Picasso-RecyclerView-StaggeredGridLayoutManager You can download and start customizing it for your project also. How to use with Picasso Picasso + RecyclerView + StaggeredGridLayoutManager Its the enhanced version of this project https://github.com/pohh/slotmachinepicasso were there was a problem with Picasso + RecyclerView + StaggeredGridLayoutManager shuffles resizable recycler views infinitely issue posted on github https://github.com/square/picasso/issues/918 I have made some changes now it works with Picasso and Glide without any shuffles and position change Currently this project is done with Picasso If you want to use it with Glide How to use with Glide Glide + RecyclerView + StaggeredGridLayoutManager Add dependencies for Glide https://github.com/bumptech/glide Remove Picasso library from dependency and remove all the codes of Picasso from MyGridAdapter.java and also from other p...