【支持動態gif格式】基於java實現圖像裁剪以及生成縮略圖功能

    通過複雜的思想鬥爭,本人作出一個艱難的決定--放棄ITEye,轉戰OSC!現發表OSC處女文,以做留念~  java

    本文主要介紹基於java實現圖像裁剪以及生成縮略圖的方法。結果代表,JDK原生圖像處理工具包(imageIO)對gif格式處理先天不足,詳細探索代碼以下: ios

import java.awt.Color;
import java.awt.Graphics;
import java.awt.Image;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.util.Arrays;
import java.util.Iterator;

import javax.imageio.IIOImage;
import javax.imageio.ImageIO;
import javax.imageio.ImageReader;
import javax.imageio.ImageWriter;
import javax.imageio.stream.ImageInputStream;
import javax.imageio.stream.ImageOutputStream;


/**
 * 圖像裁剪以及壓縮處理工具類
 *
 * 主要針對動態的GIF格式圖片裁剪以後,只出現一幀動態效果的現象提供解決方案
 *
 * 提供依賴三方包解決方案(針對GIF格式數據特徵一一解析,進行編碼解碼操做)
 * 提供基於JDK Image I/O 的解決方案(JDK探索失敗)
 *
 *
 * @author Andy
 * @see       GifDecoder.class
 * @see       AnimatedGifEncoder.class
 * @see       BufferedImage.class
 * @see    ImageIO.class
 * @see       ImageReader.class
 * @since 1.0 2011.12.21
 */
public class ImageCutterUtil {

    public enum IMAGE_FORMAT{
        BMP("bmp"),
        JPG("jpg"),
        WBMP("wbmp"),
        JPEG("jpeg"),
        PNG("png"),
        GIF("gif");
        
        private String value;
        IMAGE_FORMAT(String value){
            this.value = value;
        }
        public String getValue() {
            return value;
        }
        public void setValue(String value) {
            this.value = value;
        }
    }
    
    
    /**
     * 獲取圖片格式
     * @param file   圖片文件
     * @return    圖片格式
     */
    public static String getImageFormatName(File file)throws IOException{
        String formatName = null;
        
        ImageInputStream iis = ImageIO.createImageInputStream(file);
        Iterator<ImageReader> imageReader =  ImageIO.getImageReaders(iis);
        if(imageReader.hasNext()){
            ImageReader reader = imageReader.next();
            formatName = reader.getFormatName();
        }

        return formatName;
    }
    
    /*************************  基於三方包解決方案    *****************************/
    /**
     * 剪切圖片
     *
     * @param source        待剪切圖片路徑
     * @param targetPath    裁剪後保存路徑(默認爲源路徑)
     * @param x                起始橫座標
     * @param y                起始縱座標
     * @param width            剪切寬度
     * @param height        剪切高度         
     *
     * @returns                   裁剪後保存路徑(圖片後綴根據圖片自己類型生成)    
     * @throws IOException
     */
    public static String cutImage(String sourcePath , String targetPath , int x , int y , int width , int height) throws IOException{
        File file = new File(sourcePath);
        if(!file.exists()) {
            throw new IOException("not found the image:" + sourcePath);
        }
        if(null == targetPath || targetPath.isEmpty()) targetPath = sourcePath;
        
        String formatName = getImageFormatName(file);
        if(null == formatName) return targetPath;
        formatName = formatName.toLowerCase();
        
        // 防止圖片後綴與圖片自己類型不一致的狀況
        String pathPrefix = getPathWithoutSuffix(targetPath);
        targetPath = pathPrefix + formatName;
        
        // GIF須要特殊處理
        if(IMAGE_FORMAT.GIF.getValue() == formatName){
            GifDecoder decoder = new GifDecoder();  
            int status = decoder.read(sourcePath);  
            if (status != GifDecoder.STATUS_OK) {  
                throw new IOException("read image " + sourcePath + " error!");  
            }

            AnimatedGifEncoder encoder = new AnimatedGifEncoder();
            encoder.start(targetPath);
            encoder.setRepeat(decoder.getLoopCount());  
            for (int i = 0; i < decoder.getFrameCount(); i ++) {  
                encoder.setDelay(decoder.getDelay(i));  
                BufferedImage childImage = decoder.getFrame(i);
                BufferedImage image = childImage.getSubimage(x, y, width, height);
                encoder.addFrame(image);  
            }  
            encoder.finish();
        }else{
            BufferedImage image = ImageIO.read(file);
            image = image.getSubimage(x, y, width, height);
            ImageIO.write(image, formatName, new File(targetPath));
        }
        
        return targetPath;
    }
    
    /**
     * 壓縮圖片
     * @param sourcePath       待壓縮的圖片路徑
     * @param targetPath    壓縮後圖片路徑(默認爲初始路徑)
     * @param width            壓縮寬度
     * @param height        壓縮高度
     *
     * @returns                   裁剪後保存路徑(圖片後綴根據圖片自己類型生成)    
     * @throws IOException
     */
    public static String zoom(String sourcePath , String targetPath, int width , int height) throws IOException{
        File file = new File(sourcePath);
        if(!file.exists()) {
            throw new IOException("not found the image :" + sourcePath);
        }
        if(null == targetPath || targetPath.isEmpty()) targetPath = sourcePath;
        
        String formatName = getImageFormatName(file);
        if(null == formatName) return targetPath;
        formatName = formatName.toLowerCase();
        
        // 防止圖片後綴與圖片自己類型不一致的狀況
        String pathPrefix = getPathWithoutSuffix(targetPath);
        targetPath = pathPrefix + formatName;
        
        // GIF須要特殊處理
        if(IMAGE_FORMAT.GIF.getValue() == formatName){
            GifDecoder decoder = new GifDecoder();  
            int status = decoder.read(sourcePath);  
            if (status != GifDecoder.STATUS_OK) {  
                throw new IOException("read image " + sourcePath + " error!");  
            }

            AnimatedGifEncoder encoder = new AnimatedGifEncoder();
            encoder.start(targetPath);
            encoder.setRepeat(decoder.getLoopCount());  
            for (int i = 0; i < decoder.getFrameCount(); i ++) {  
                encoder.setDelay(decoder.getDelay(i));  
                BufferedImage image = zoom(decoder.getFrame(i), width , height);
                encoder.addFrame(image);  
            }  
            encoder.finish();
        }else{
            BufferedImage image = ImageIO.read(file);
            BufferedImage zoomImage = zoom(image , width , height);
            ImageIO.write(zoomImage, formatName, new File(targetPath));
        }
        
        return targetPath;
    }
    
    /*********************** 基於JDK 解決方案     ********************************/
    
    /**
     * 讀取圖片
     * @param file 圖片文件
     * @return     圖片數據
     * @throws IOException
     */
    public static BufferedImage[] readerImage(File file) throws IOException{
        BufferedImage sourceImage = ImageIO.read(file);
        BufferedImage[] images = null;
        ImageInputStream iis = ImageIO.createImageInputStream(file);
        Iterator<ImageReader> imageReaders = ImageIO.getImageReaders(iis);
        if(imageReaders.hasNext()){
            ImageReader reader = imageReaders.next();
            reader.setInput(iis);
            int imageNumber = reader.getNumImages(true);
            images = new BufferedImage[imageNumber];
            for (int i = 0; i < imageNumber; i++) {
                BufferedImage image = reader.read(i);
                if(sourceImage.getWidth() > image.getWidth() || sourceImage.getHeight() > image.getHeight()){
                    image = zoom(image, sourceImage.getWidth(), sourceImage.getHeight());
                }
                images[i] = image;
            }
            reader.dispose();
            iis.close();
        }
        return images;
    }
    
    /**
     * 根據要求處理圖片
     *
     * @param images    圖片數組
     * @param x            橫向起始位置
     * @param y         縱向起始位置
     * @param width      寬度    
     * @param height    寬度
     * @return            處理後的圖片數組
     * @throws Exception
     */
    public static BufferedImage[] processImage(BufferedImage[] images , int x , int y , int width , int height) throws Exception{
        if(null == images){
            return images;
        }
        BufferedImage[] oldImages = images;
        images = new BufferedImage[images.length];
        for (int i = 0; i < oldImages.length; i++) {
            BufferedImage image = oldImages[i];
            images[i] = image.getSubimage(x, y, width, height);
        }
        return images;
    }
    
    /**
     * 寫入處理後的圖片到file
     *
     * 圖片後綴根據圖片格式生成
     *
     * @param images        處理後的圖片數據
     * @param formatName     圖片格式
     * @param file            寫入文件對象
     * @throws Exception
     */
    public static void writerImage(BufferedImage[] images ,  String formatName , File file) throws Exception{
        Iterator<ImageWriter> imageWriters = ImageIO.getImageWritersByFormatName(formatName);
        if(imageWriters.hasNext()){
            ImageWriter writer = imageWriters.next();
            String fileName = file.getName();
            int index = fileName.lastIndexOf(".");
            if(index > 0){
                fileName = fileName.substring(0, index + 1) + formatName;
            }
            String pathPrefix = getFilePrefixPath(file.getPath());
            File outFile = new File(pathPrefix + fileName);
            ImageOutputStream ios = ImageIO.createImageOutputStream(outFile);
            writer.setOutput(ios);
            
            if(writer.canWriteSequence()){
                writer.prepareWriteSequence(null);
                for (int i = 0; i < images.length; i++) {
                    BufferedImage childImage = images[i];
                    IIOImage image = new IIOImage(childImage, null , null);
                    writer.writeToSequence(image, null);
                }
                writer.endWriteSequence();
            }else{
                for (int i = 0; i < images.length; i++) {
                    writer.write(images[i]);
                }
            }
            
            writer.dispose();
            ios.close();
        }
    }
    
    /**
     * 剪切格式圖片
     *
     * 基於JDK Image I/O解決方案
     *
     * @param sourceFile        待剪切圖片文件對象
     * @param destFile                  裁剪後保存文件對象
     * @param x                    剪切橫向起始位置
     * @param y                 剪切縱向起始位置
     * @param width              剪切寬度    
     * @param height            剪切寬度
     * @throws Exception
     */
    public static void cutImage(File sourceFile , File destFile, int x , int y , int width , int height) throws Exception{
        // 讀取圖片信息
        BufferedImage[] images = readerImage(sourceFile);
        // 處理圖片
        images = processImage(images, x, y, width, height);
        // 獲取文件後綴
        String formatName = getImageFormatName(sourceFile);
        destFile = new File(getPathWithoutSuffix(destFile.getPath()) + formatName);

        // 寫入處理後的圖片到文件
        writerImage(images, formatName , destFile);
    }
    
    
    
    /**
     * 獲取系統支持的圖片格式
     */
    public static void getOSSupportsStandardImageFormat(){
        String[] readerFormatName = ImageIO.getReaderFormatNames();
        String[] readerSuffixName = ImageIO.getReaderFileSuffixes();
        String[] readerMIMEType = ImageIO.getReaderMIMETypes();
        System.out.println("========================= OS supports reader ========================");
        System.out.println("OS supports reader format name :  " + Arrays.asList(readerFormatName));
        System.out.println("OS supports reader suffix name :  " + Arrays.asList(readerSuffixName));
        System.out.println("OS supports reader MIME type :  " + Arrays.asList(readerMIMEType));
        
        String[] writerFormatName = ImageIO.getWriterFormatNames();
        String[] writerSuffixName = ImageIO.getWriterFileSuffixes();
        String[] writerMIMEType = ImageIO.getWriterMIMETypes();
        
        System.out.println("========================= OS supports writer ========================");
        System.out.println("OS supports writer format name :  " + Arrays.asList(writerFormatName));
        System.out.println("OS supports writer suffix name :  " + Arrays.asList(writerSuffixName));
        System.out.println("OS supports writer MIME type :  " + Arrays.asList(writerMIMEType));
    }
    
    /**
     * 壓縮圖片
     * @param sourceImage    待壓縮圖片
     * @param width             壓縮圖片高度
     * @param heigt            壓縮圖片寬度
     */
    private static BufferedImage zoom(BufferedImage sourceImage , int width , int height){
        BufferedImage zoomImage = new BufferedImage(width, height, sourceImage.getType());
        Image image = sourceImage.getScaledInstance(width, height, Image.SCALE_SMOOTH);
        Graphics gc = zoomImage.getGraphics();
        gc.setColor(Color.WHITE);
        gc.drawImage( image , 0, 0, null);
        return zoomImage;
    }
    
    /**
     * 獲取某個文件的前綴路徑
     *
     * 不包含文件名的路徑
     *
     * @param file   當前文件對象
     * @return
     * @throws IOException
     */
    public static String getFilePrefixPath(File file) throws IOException{
        String path = null;
        if(!file.exists()) {
            throw new IOException("not found the file !" );
        }
        String fileName = file.getName();
        path = file.getPath().replace(fileName, "");
        return path;
    }
    
    /**
     * 獲取某個文件的前綴路徑
     *
     * 不包含文件名的路徑
     *
     * @param path   當前文件路徑
     * @return         不包含文件名的路徑
     * @throws Exception
     */
    public static String getFilePrefixPath(String path) throws Exception{
        if(null == path || path.isEmpty()) throw new Exception("文件路徑爲空!");
        int index = path.lastIndexOf(File.separator);
        if(index > 0){
            path = path.substring(0, index + 1);
        }
        return path;
    }
    
    /**
     * 獲取不包含後綴的文件路徑
     *
     * @param src
     * @return
     */
    public static String getPathWithoutSuffix(String src){
        String path = src;
        int index = path.lastIndexOf(".");
        if(index > 0){
            path = path.substring(0, index + 1);
        }
        return path;
    }
    
    /**
     * 獲取文件名
     * @param filePath        文件路徑
     * @return                文件名
     * @throws IOException
     */
    public static String getFileName(String filePath) throws IOException{
        File file = new File(filePath);
        if(!file.exists()) {
            throw new IOException("not found the file !" );
        }
        return file.getName();
    }
    
    /**
     * @param args
     * @throws Exception
     */
    public static void main(String[] args) throws Exception {
        // 獲取系統支持的圖片格式
        //ImageCutterUtil.getOSSupportsStandardImageFormat();
        
        try {
            // 起始座標,剪切大小
            int x = 100;
            int y = 75;
            int width = 100;
            int height = 100;
            // 參考圖像大小
            int clientWidth = 300;
            int clientHeight = 250;
            
            
            File file = new File("D:\\PCM Project\\upload\\tmp\\1.gif");
            BufferedImage image = ImageIO.read(file);
            double destWidth = image.getWidth();
            double destHeight = image.getHeight();
            
            if(destWidth < width || destHeight < height)
                throw new Exception("源圖大小小於截取圖片大小!");
            
            double widthRatio = destWidth / clientWidth;
            double heightRatio = destHeight / clientHeight;
            
            x = Double.valueOf(x * widthRatio).intValue();
            y = Double.valueOf(y * heightRatio).intValue();
            width = Double.valueOf(width * widthRatio).intValue();
            height = Double.valueOf(height * heightRatio).intValue();
            
            System.out.println("裁剪大小  x:" + x + ",y:" + y + ",width:" + width + ",height:" + height);

            /************************ 基於三方包解決方案 *************************/
            String formatName = getImageFormatName(file);
            String pathSuffix = "." + formatName;
            String pathPrefix = getFilePrefixPath(file);
            String targetPath = pathPrefix  + System.currentTimeMillis() + pathSuffix;
            targetPath = ImageCutterUtil.cutImage(file.getPath(), targetPath, x , y , width, height);
            
            String bigTargetPath = pathPrefix  + System.currentTimeMillis() + pathSuffix;
            ImageCutterUtil.zoom(targetPath, bigTargetPath, 100, 100);
            
            String smallTargetPath = pathPrefix  + System.currentTimeMillis() + pathSuffix;
            ImageCutterUtil.zoom(targetPath, smallTargetPath, 50, 50);
            
            /************************ 基於JDK Image I/O 解決方案(JDK探索失敗) *************************/
//            File destFile = new File(targetPath);
//            ImageCutterUtil.cutImage(file, destFile, x, y, width, height);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
相關文章
相關標籤/搜索