2

I'm working with a Base64 Image, and I have to resize that one, the code that I'm using now, in some way is doing this work. But I have notice that I'm losing quality and the image doesn't has the size that I set. I'm not sure what the reason is and how can I fix it.

For example the original image looks like this:

enter image description here

But after the resize the image lost a lot of quality and the final image looks like this:

enter image description here

This is the code I'm using:

import java.awt.*;
import java.io.IOException;

import org.opencv.core.Mat;
import sun.misc.BASE64Encoder;
import sun.misc.BASE64Decoder;

import javax.imageio.ImageIO;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.awt.image.BufferedImage;


public class Base64ImageUtil {
    private static final String JPG = "";

    /**
     * Decode string to image
     *
     * @param imageString The string to decode
     * @return decoded image
     */
    public static BufferedImage decodeToImage(String imageString) throws IOException {
        BufferedImage image = null;
        byte[] imageByte;
        BASE64Decoder decoder = new BASE64Decoder();
        imageByte = decoder.decodeBuffer(imageString);
        ByteArrayInputStream bis = new ByteArrayInputStream(imageByte);
        image = ImageIO.read(bis);
        bis.close();
        return image;
    }

    /**
     * Encode image to string
     *
     * @param imageFile The image to encode
     * @param type      jpeg, bmp, gif, png
     * @return encoded string
     */
    public static String encodeToString(File imageFile, String type) throws IOException {
        String imageString = null;
        ByteArrayOutputStream bos = new ByteArrayOutputStream();

        BufferedImage image = ImageIO.read(imageFile);
        ImageIO.write(image, type, bos);
        byte[] imageBytes = bos.toByteArray();

        BASE64Encoder encoder = new BASE64Encoder();
        imageString = encoder.encode(imageBytes);

        bos.close();
        return imageString;
    }

    /**
     * To make base64 string decoded properly, We need to remove the base64 header from a base64 string.
     *
     * @param base64 The Base64 string of an image.
     * @return Base64 string without header.
     */
    public static String removeBase64Header(String base64) {
        if (base64 == null) return null;
        return base64.trim().replaceFirst("data[:]image[/]([a-z])+;base64,", "");
    }

    /**
     * Get the image type.
     *
     * @param base64 The base64 string of an image.
     * @return jpg, png, gif
     */
    public static String getImageType(String base64) {
        String[] header = base64.split("[;]");
        if (header == null || header.length == 0) return null;
        return header[0].split("[/]")[1];
    }

    public static BufferedImage resize(BufferedImage img, int newW, int newH) {
        Image tmp = img.getScaledInstance(newW, newH, Image.SCALE_REPLICATE);
        BufferedImage dimg = new BufferedImage(newW, newH, BufferedImage.TYPE_INT_ARGB);

        Graphics2D g2d = dimg.createGraphics();
        g2d.drawImage(tmp, 0, 0, null);
        g2d.dispose();

        return dimg;
    }

    public static void main(String args[]) throws IOException {
        String base64 = JPG;
        // convert base64 string to image
        String imageString = removeBase64Header(base64);
        String imageType = getImageType(base64);
        BufferedImage image = decodeToImage(imageString);
        File imageFile = File.createTempFile("image", "." + imageType);
        image = resize(image, 600, 800);
        ImageIO.write(image, imageType, imageFile);
        // convert image to base64 string
        String newImageString = encodeToString(imageFile, imageType);
        System.out.println("Image Type -> " + imageType);
        System.out.println(newImageString);
        System.out.println();
        System.out.println();
    }

    public static BufferedImage mat2Img(Mat in)
    {
        BufferedImage out;
        byte[] data = new byte[320 * 240 * (int)in.elemSize()];
        int type;
        in.get(0, 0, data);

        if(in.channels() == 1)
            type = BufferedImage.TYPE_BYTE_GRAY;
        else
            type = BufferedImage.TYPE_3BYTE_BGR;

        out = new BufferedImage(320, 240, type);

        out.getRaster().setDataElements(0, 0, 320, 240, data);
        return out;
    }

}
rasilvap
  • 1,771
  • 3
  • 31
  • 70
  • I found this, do not know if it helps https://stackoverflow.com/questions/1922040/how-to-resize-an-image-c-sharp – Maytham Fahmi Sep 04 '19 at 21:04
  • 1) You are not maintaining the aspect ratio of the image when scaling, there is an obvious stretch on the vertical axis. 2) You're not setting the correct image type so the colors are off, basically `BufferedImage.TYPE_INT_ARGB` should be changed (to what I do not know). – Jake Holzinger Sep 04 '19 at 21:13
  • 1
    @JakeHolzinger I believe just using `TYPE_INT_RGB` would be an improvement. Alternatively, `TYPE_3BYTE_BGR`. Both should allow you to store the output as a JPEG with correct colors and no fuss. – Harald K Sep 05 '19 at 12:18
  • 1
    Also, using `Image.getScaledInstance(w, h, Image.SCALE_REPLICATE)` (i.e. "nearest neighbour") is not what you want, if quality is your quest... `SCALE_SMOOTH` or `SCALE_AREA_AVERAGING` is better, but for best results, use a library like imgScalr or a similar, high quality algorithm. – Harald K Sep 05 '19 at 12:29
  • @JakeHolzinger, thanks it worked! – rasilvap Sep 05 '19 at 15:24

1 Answers1

0

I had done this little change:

BufferedImage dimg = new BufferedImage(newW, newH, BufferedImage.TYPE_USHORT_555_RGB);

And use TYPE_USHORT_555_RGB it worked for me.

rasilvap
  • 1,771
  • 3
  • 31
  • 70