struts2+jsp+jquery+Jcrop实现图片裁剪并上传

<1> 使用html标签上传需要裁剪的大图。
<2> 在页面呈现大图,使用Jcrop(Jquery)对大图进行裁剪,并且可以进行预览。
<3> 选择好截取部分之后发送数据给Action,在服务器端使用 Java API 对大图进行裁剪。
<4> 保存大图裁剪好的头像到指定目录,完成业务。

下面一步一步做:

第一步:使用html标签上传需要裁剪的大图。

这一步说白了也就是使用Struts2自带的FileUpload功能,把图片进行上传具体代码如下:
html页面:

[html] view plaincopy
<form id="ulform" action="uploadPic.action" enctype="multipart/form-data" method="post">
    <input type="file" name="pic" id="file" value="选择图片" />
    <input type="submit" value="点击上传" />
</form>  

Struts2配置文件
[html] view plaincopy
<action name="uploadPic" class="com.luoxiao.tbms.user.action.UserAction" method="uploadPic">
    <result name="success" type="redirect">changePic.jsp</result>
    <result name="error">changePic.jsp</result>
</action>  

根据配置点击提交按钮,会提交表单,把图片以流的形式发送给 UserAction的uploadPic方法,该方法如下:

[java] view plaincopy
public class UserAction{
    private File pic;   //(在此省略 get 和 set 方法)
    private String picFileName; //(省略get和set方法, 该属性Struts2会自动赋值为上传文件的文件名)
    public String uploadPic() {
        String[] str = { ".jpg", ".jpeg", ".bmp", ".gif" };
        // 获取用户登录名
        TbUser curruser = (TbUser) getValue(SCOPE_SESSION, "curruser");
        // 限定文件大小是4MB
        if (pic == null || pic.length() > 4194304) {
            //文件过大
            return "error";
        }
        for (String s : str) {
            if (picFileName.endsWith(s)) {
                String realPath = ServletActionContext.getServletContext().getRealPath("/uploadpic");// 在tomcat中保存图片的实际路径  ==  "webRoot/uploadpic/"
                File saveFile = new File(new File(realPath), "新文件名.jpg"); // 在该实际路径下实例化一个文件
                // 判断父目录是否存在
                if (!saveFile.getParentFile().exists()) {
                    saveFile.getParentFile().mkdirs();
                }
                try {
                    // 执行文件上传
                    // FileUtils 类名 org.apache.commons.io.FileUtils;
                    // 是commons-io包中的,commons-fileupload 必须依赖
                    // commons-io包实现文件上次,实际上就是将一个文件转换成流文件进行读写
                    FileUtils.copyFile(pic, saveFile);
                } catch (IOException e) {
                    return "imageError";
                }
            }
        }
        return "success";
    }
}  

这样就可以把用户选择的图片上传到tomcat的webRoot/uploadpic/文件夹下。 然后访问页面,页面中就可以显示出刚刚上传的大图了。代码如下。

[html] view plaincopy
<div style="width: 500px; height: 500px;">
    <img style="margin-top:20px;" src="../uploadpic/上传文件名称.jpg"/>" id="target"  />
</div>
第一步完成。

第二步:使用Jcrop插件裁剪该图片,并且在页面中预览。

Jcrop是一个基于JQuery的成熟的图片裁剪的插件。如图:

想要使用它,需要去官方网站下载该js,现给出一个地址:(附带jquery)

点击去往Jcrop插件下载页面

该插件使用比较简单:

<1> 在裁剪图片页面中,引入两个js文件,和1个Jcrop需要的css文件(Jcrop包中有,注意引入顺序,先引入jquery):
[html] view plaincopy
<script src="../js/jquery-1.8.3.min.js" type="text/javascript"></script>
<script src="../js/jquery.Jcrop.js" type="text/javascript"></script>
<link rel="stylesheet" href="../css/jquery.Jcrop.css" type="text/css" />  

<2> 在html页面中按照Jcrop要求的格式编写两个img标签,一个用作裁剪后的预览,一个用作显示大图,代码如下:

[html] view plaincopy
预览:
<div style="width:200px;height:200px;overflow:hidden; border:1px solid gray;">
   <img id="preview" width="200px" height="200px" />
</div>
原图:
   <img src="../uploadpic/上传大图.jpg" id="target"  />  

<3> 在该页面中写js代码,使其可以裁剪图片并且预览:

[javascript] view plaincopy
<script type="text/javascript">
        var x;
    var y;
    var width;
    var height;
    $(function(){
        var jcrop_api, boundx, boundy;
        <span style="color:#ff0000;">//使原图具有裁剪功能</span>
        $(‘#target‘).Jcrop({
            onChange: updatePreview,
            onSelect: updatePreview,
            aspectRatio: 1
        },function(){
            // Use the API to get the real image size
            var bounds = this.getBounds();
            boundx = bounds[0];
            boundy = bounds[1];
            // Store the API in the jcrop_api variable
            jcrop_api = this;
        });
        <span style="color:#ff0000;">//裁剪过程中,每改变裁剪大小执行该函数</span>
        function updatePreview(c){
            if (parseInt(c.w) > 0){
                $(‘#preview‘).css({
                    width: Math.round(<span style="color:#ff0000;">200 </span>/ c.w * boundx) + ‘px‘,   <span style="color:#ff0000;">//200 为预览div的宽和高</span>
                    height: Math.round(<span style="color:#ff0000;">200 </span>/ c.h * boundy) + ‘px‘,
                    marginLeft: ‘-‘ + Math.round(200 / c.w * c.x) + ‘px‘,
                    marginTop: ‘-‘ + Math.round(200 / c.h * c.y) + ‘px‘
                });
                <span style="color:#ff0000;">$(‘#width‘).val(c.w);  //c.w 裁剪区域的宽
                $(‘#height‘).val(c.h); //c.h 裁剪区域的高
                $(‘#x‘).val(c.x);  //c.x 裁剪区域左上角顶点相对于图片左上角顶点的x坐标
                $(‘#y‘).val(c.y);  //c.y 裁剪区域顶点的y坐标</span>
            }
          };
    });
  </script>  

至此我们已经可以看到裁剪之后的样子了,并且也可以得到裁剪区域的x,y,height,width属性。

第三步:把截取的该区域的属性传递给action,让action根据所得属性,利用javaAPI把原图裁剪成小图。

<1> 设置form表单与隐藏域表单组件,并且在裁剪的时候对该四个组件的value属性赋值

[html] view plaincopy
<form action="cutPic.action" method="post">
    点击
    <input type="hidden" name="image.x" id="x"/>
    <input type="hidden" name="image.y" id="y"/>
    <input type="hidden" name="image.width" id="width"/>
    <input type="hidden" name="image.height" id="height"/>
    <input type="submit" value="确定" />
    ,设置完成。
</form>  

<2> 点击确定,提交该表单,访问action,配置如下:

[html] view plaincopy
<action name="cutPic" class="com.luoxiao.tbms.user.action.UserAction" method="cutPic">
    <result name="success" type="redirectAction">../announcement/announcement_list.action</result>
</action>  

<3>Struts2带着四个参数访问UserAction,并且会自动给UserAction中的image属性赋值,该image属性为OperateImage的一个实例对象,该类为裁剪图片类,代码如下:

[java] view plaincopy
package com.luoxiao.util;  

import java.awt.Rectangle;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Iterator;
import javax.imageio.ImageIO;
import javax.imageio.ImageReadParam;
import javax.imageio.ImageReader;
import javax.imageio.stream.ImageInputStream;  

public class OperateImage {
    // ===源图片路径名称如:c:\1.jpg
    private String srcpath;
    // ===剪切图片存放路径名称.如:c:\2.jpg
    private String subpath;
    // ===剪切点x坐标
    private int x;
    private int y;
    // ===剪切点宽度
    private int width;
    private int height;   <span style="color:#ff0000;">(省略四个属性的get & set)</span>
    public OperateImage() {
    }
    /** 对图片裁剪,并把裁剪完的新图片保存 */
    public void cut() throws IOException {
        FileInputStream is = null;
        ImageInputStream iis = null;
        try {
            // 读取图片文件
            is = new FileInputStream(srcpath);
            /*
             * 返回包含所有当前已注册 ImageReader 的 Iterator,这些 ImageReader 声称能够解码指定格式。
             * 参数:formatName - 包含非正式格式名称 . (例如 "jpeg" 或 "tiff")等 。
             */
            Iterator<ImageReader> it = ImageIO.getImageReadersByFormatName("jpg");
            ImageReader reader = it.next();
            // 获取图片流
            iis = ImageIO.createImageInputStream(is);
            /*
             * <p>iis:读取源.true:只向前搜索 </p>.将它标记为 ‘只向前搜索’。
             * 此设置意味着包含在输入源中的图像将只按顺序读取,可能允许 reader 避免缓存包含与以前已经读取的图像关联的数据的那些输入部分。
             */
            reader.setInput(iis, true);
            /*
             * <p>描述如何对流进行解码的类<p>.用于指定如何在输入时从 Java Image I/O
             * 框架的上下文中的流转换一幅图像或一组图像。用于特定图像格式的插件 将从其 ImageReader 实现的
             * getDefaultReadParam 方法中返回 ImageReadParam 的实例。
             */
            ImageReadParam param = reader.getDefaultReadParam();
            /*
             * 图片裁剪区域。Rectangle 指定了坐标空间中的一个区域,通过 Rectangle 对象
             * 的左上顶点的坐标(x,y)、宽度和高度可以定义这个区域。
             */
            Rectangle rect = new Rectangle(x, y, width, height);
            // 提供一个 BufferedImage,将其用作解码像素数据的目标。
            param.setSourceRegion(rect);
            /*
             * 使用所提供的 ImageReadParam 读取通过索引 imageIndex 指定的对象,并将 它作为一个完整的
             * BufferedImage 返回。
             */
            BufferedImage bi = reader.read(0, param);
            // 保存新图片
            ImageIO.write(bi, "jpg", new File(subpath));
        } finally {
            if (is != null)
                is.close();
            if (iis != null)
                iis.close();
        }  

    }
}  

<4> 给该类的实例的四个属性 x,y,width,height赋值之后,访问action中的cutPic方法,代码如下:

[java] view plaincopy
public class UserAction extends BaseAction {
    private OperateImage image;(省略get set)  

    private File pic; // 接收这个上传的文件
    private String picFileName; // Struts2提供的格式,在文件名后+FileName就是上传文件的名字  

    /**
     * 裁剪头像
     */
    public String cutPic(){
<span style="color:#ff0000;">       String name = ServletActionContext.getServletContext().getRealPath("/uploadpic/原图名.jpg");
        image.setSrcpath(name);
        image.setSubpath(ServletActionContext.getServletContext().getRealPath("/uploadpic/裁剪目标图名.jpg"));
        try {
            image.cut(); //执行裁剪操作  执行完后即可生成目标图在对应文件夹内。</span>
        } catch (IOException e) {
            e.printStackTrace();
        }
        return "success";
    }
}  

第四步:把截取好的头像保存在具体文件夹下即可,裁剪过程完成。

  

时间: 2024-08-18 18:48:59

struts2+jsp+jquery+Jcrop实现图片裁剪并上传的相关文章

基于JSP+Servlet+JavaBean的图片或文件上传

基于JSP+Servlet+JavaBean的图片或文件上传 一.概述 现在不管是博客论坛还是企业办公,都离不开资源的共享.通过文件上传的方式,与大家同分享,从而达到大众间广泛的沟通和交流,我们既可以从中获得更多的知识和经验,也能通过他人的反馈达到自我改进和提升的目的. 下面我就为大家介绍 web项目中的这一上传功能,那么文件是如何从本地发送到服务器的呢?大家可以在在线视频课程进修学习<基于JSP+Servlet+JavaBean的人力资源管理系统开发>中第22课-项目开发-其它功能完善-图片

前台Jcrop配合后台Graphics实现图片裁剪并上传

Jcrop:一个功能强大的图片裁剪插件 版本:Jcrop v0.9.12 VS版本:vs2015 下载地址:http://code.ciaoca.com/jquery/jcrop/version/Jcrop-0.9.12.zip 本文主要讲的是,在前台通过file选择图片,然后用Jcrop裁剪好图片后,把偏移量等参数传值到后台,在后台通过Graphics进行图片的截取并上传到服务器,显示一下重点代码 HTML 部分<div class="example"> <img

file API+JCrop 选择、裁剪并上传头像

file API,直接读取本地文件,绕过了后台处理再返回前台的过程,使编程更简单.fileAPI和JCrop的结合使用上传头像. 1.html部分 <input id="demo" type="file" onchange="showPic();"/> <img id="avatar" src=""> 2.fileAPI预览图片(jquery) <script type=&qu

jquery.uploadView 实现图片预览上传

图片上传,网上有好多版本,今天也要做一个查了好多最终找到了一个uploadview 进行了一下修改 来看代码 @{ Layout = null; } <!DOCTYPE html> <html> <head> <meta name="viewport" content="width=device-width" /> <title>Index</title> <script src=&quo

android 用户头像,图片裁剪,上传并附带用户数据base64code 方式

图片上传的文件流我上一篇博客写了,这一篇我们说一下base64,base64上传方式就是将图片转换成base64码,然后把base64码以字符串的方式上传,然后服务器接收到以后再解码就可以了,相对于文件流来说比较简单: 用户头像上传我们首先要获得图片的url然后再裁剪图片,然后把裁剪后的图片转换成base64然后在上传: 下边是安卓端代码: 首先我们要获得裁剪后的图片:一,选择图片: 代码如下,通过对话框选择获得图片的方式: activity: /* * 提示对话框 */ private voi

基于jQuery移动设备图片裁剪代码

一款基于jQuery移动设备图片裁剪代码.这是一款支持移动设备触摸手势的jQuery图片裁剪插件jquery.photoClip.效果图如下: 在线预览   源码下载 实现的代码. html代码: <article class="zzsc-container"> <div id="clipArea"></div> <input type="file" id="file"> <

JSP Ueditor 实现图片跨域上传

Ueditor的单图上传,在官方文档上明确写了不支持 然后通过百度找了许多方案,终于有一个可以解决了. http://www.cnblogs.com/hpnet/p/6290452.html 不过那个是.NET版本的,按他的方案,自己写了一个JAVA版本的. 核心思路就是把数据重发到原地址. 所以在原地址控制层写中转代码: 修改Ueditor中的controller.jsp中的代码,在获取上传数据到,再跳转到原地址界面:

PHP+ajaxfileupload与jcrop插件结合 完成头像上传

昨天花了点时间整合了一下头像插件 东拼西凑的成果 先来看下效果 1.先使用ajaxfileupload插件做异步上传.这个地方我本来想做个上传进度的效果,但技术有限失败了.上传按钮我还做了一个文件大小的限制,但是由于浏览器兼容性的问题,不完美在IE6--IE9之间还有很多问题需要解决 getFileSize函数是用于判断文件大小的函数 function getFileSize(fileName) { var byteSize = 0; //console.log($("#" + fil

jQuery+php实现ajax文件即时上传

很多项目中需要用到即时上传功能,比如,选择本地图片后,立即上传并显示图像.本文结合实例讲解如何使用jQuery和PHP实现Ajax即时上传文件的功能,用户只需选择本地图片确定后即实现上传,并显示上传进度条,上传完成后,显示图片信息. 查看演示DEMO下载源码 HTML 本示例基于jQuery以及相当出色的jquery.form插件,所以,先要载入jquery库和form插件.  <script type="text/javascript" src="jquery.min