分享我自己弄的图片压缩上传的DEMO,带服务器端。用H5+的拍照和选择相册功能都可以上传。

我是参照官方推荐的js---->localResizeIMG改的,这个多看看就懂了。

这个是版本2--->localResizeIMG-2导入JS,直接丢一个页面里就能看到效果。代码很简洁,就是我看不懂。。大家可以看看挺厉害的。

具体就是用HTML 5 <canvas> 处理图片,再生成Base64编码。然后把这段Base64传到服务器端解码-写入文件就行啦。后面会说一下!

图片的选择用H5 的拍照或者选择相册都行,就像下面

// 拍照添加文件

function getImage() {

var cmr = plus.camera.getCamera();

cmr.captureImage( function ( p ) {

plus.io.resolveLocalFileSystemURL( p, function ( entry ) {

if(files.length<2){

var localurl = entry.toLocalURL();//把拍照的目录路径,变成本地url路径,例如file:///........之类的。

appendFile(localurl);

}

});

},function( error ) {

alert( "Capture image failed: " error.message );

}

);

}



// 从相册添加文件

function appendByGallery(){

plus.gallery.pick(function(path){



appendFile(path);//处理图片的地方

});

}



\n

<----------------------------------接下来是处理图片------------------------------------------------>

// 添加文件

var f1=null;

function appendFile(path){



var img = new Image();

img.src = path; // 传过来的图片路径在这里用。

img.onload = function () {

var that = this;

//生成比例

var w = that.width,

h = that.height,

scale = w / h;

w = 480 || w; //480 你想压缩到多大,改这里

h = w / scale;



//生成canvas

var canvas = document.createElement('canvas');



var ctx = canvas.getContext('2d');



$(canvas).attr({width : w, height : h});



ctx.drawImage(that, 0, 0, w, h);



var base64 = canvas.toDataURL('image/jpeg', 1 || 0.8 ); //1最清晰,越低越模糊。有一点不清楚这里明明设置的是jpeg。弹出 base64 开头的一段 data:image/png;却是png。哎开心就好,开心就好

// alert(base64);



f1 =base64; // 把base64数据丢过去,上传要用。



var pic = document.getElementById("x");

pic.src = base64; //这里丢到img 的 src 里面就能看到效果了

}



}

\n

<----------------------------------然后是上传到服务器------------------------------------------------>

// 上传文件

function upload(){



var wt=plus.nativeUI.showWaiting();

var url = '后台地址';

var dataType ='json';

//发送数据

var data = {

files1:f1 //base64数据

};

$.post(url, data, success, dataType);

}

//成功响应的回调函数

var success = function(response) {

plus.nativeUI.closeWaiting();

if(response != null){

alert("上传成功");

}



}

\n

<----------------------------------服务器接到数据,就很简单了------------------------------------------------>

<?php 





$s=dirname(__FILE__); //获的服务器路劲

$time =time(); //获得当前时间戳

$files =$_POST['files1'];

$files1 = substr($files1,22); //百度一下就可以知道base64前面一段需要清除掉才能用。

//解码

$tmp = base64_decode($files1);

$fp=$s."/uploads/".$time.".jpg"; //确定图片文件位置及名称

//写文件

file_put_contents( $fp, $tmp); //给图片文件写入数据

echo 1;

\n

大概就是这样了,写的很乱。3M的图片到服务器上就500KB左右了,这里的500Kb是自己控制的不是固定只能压缩成500kb,你可以压缩到100kb的。

\n

关于 拍摄后图片可能会方向不对,可以看楼下评论里面 --- 伟子 ---- 的回复他有提供Demo。



2014-12-31 17:38 1 条评论 分享
已邀请:
3

萌大叔 - 北航研究院项目经理

赞同来自: 多串君 cj melodic

现在插件地址已经更新到localResizeIMG3
原来的库在安卓上照片会有转动90度的问题,今天早上我已经修正,并提交给作者了,现在已经合并了代码。
大家可以下载过来测试一下

            // 从相册添加文件 

function pickUpFromGallery() {
plus.gallery.pick(function(path) {
plus.io.resolveLocalFileSystemURL(path, function(entry){
var localURL = entry.toLocalURL();
//document.getElementById('resImg').src = localURL;
compressImg(localURL); //处理图片的地方
}, function(e){
console.log("失败:" + error.message);
}, {
filename: "_doc/camera/",
index: 1
});
});
}

function compressImg(path) {
console.log('开始压缩:' + path);
lrz(path, {width:300}, function(result) {
document.getElementById('compressedImg').src = result.base64;
console.log('文件大小:' + (Math.floor(result.base64Len / 1024)) + 'K');
});
}
\n
1

伟子

赞同来自: 去你个小饼干

你这个拍照在iphone里面就不生效了哦,应该这样就可以了

var cmr = plus.camera.getCamera();
cmr.captureImage(function(p) {
plus.io.resolveLocalFileSystemURL(p, function(entry) {
appendFile(entry.fullPath); //处理图片的地方
}, function(e) {
plus.nativeUI.alert(e.message);
});
}, function(e) {}, {
filename: "_doc/camera/"
});
\n
1

C#服务器端



string image1Name = WebHelper.GetRequest("Image1Name", string.Empty);
string image1Data = WebHelper.GetRequest("Image1Data", string.Empty);
if (!String.IsNullOrEmpty(image1Name) && !String.IsNullOrEmpty(image1Data) && image1Data.Length > 23)
{
image1Data = image1Data.Substring(23);

//大图
string path = UIHelper.GetAbsoultFolder(UIHelper.GetUploadRelativePath() + image1Name);
ImageProcessor.SaveImageFromBase64String(image1Data, path);

}
\n
0

小闹

赞同来自:

谢谢分享,好东西要顶。


0

伟子

赞同来自:

不错的代码,很有用。


0

llyzlc

赞同来自:

多张图片怎么办。。。遍历for循环好像不起效。。


0

伟子

赞同来自:

还有防止图片上传到服务器不是正着的,应该在Canvas处理一下,判断图片拍摄时的角度,我是用canvasResize做的


0

llyzlc

赞同来自:

多张图片怎么办啊,又不好循环添加data数据。。只能写死么
var data = {
numimg: uploadImgArr.length.toString(),
subject: document.getElementById("key").value,
author: returnItem("useracc"),
authorid: returnItem("uid"),
sid: returnItem("sessionid"),
message: document.getElementById("message").value,
img1: uploadImgArr[0].pic,
img2: uploadImgArr[1].pic,
img3: uploadImgArr[2].pic
};


0

aaaa

赞同来自:

你这个为什么后台接收不到数据,提示undefined index club


0

h5_学习者

赞同来自:

为什么我用你的代码,在苹果手机上图片的方向会转90度啊,求解?


0

自由森

赞同来自:

500K 太大了


0

ggtiger

赞同来自:

500K 太大了 流量伤不起

如果可以设置图片质量,大小缩小在100kb以内就好了。


0

hanxin

赞同来自:

html5画布不给力啊,一张图片500k 1920*1080 压缩后更大了! 用的方法 一样:

ctx.drawImage(img,0,0,1920,1080,0,0,destw,desth);

用C#就能做到几十k,还清晰,靠,html5 怎么搞的这么挫!


0

hanxin

赞同来自:

我都可以保存为jpg啊


0

太阳光

赞同来自:

$files =$_POST['files1'];
$files1 = substr($files1,22); //这代码坑了不会PHP的同学


0

ggtiger

赞同来自:

我压缩成20kb都能看。


0

anke

赞同来自:

在iphone js压缩图片有问题,encode 有大小限制,我在iPhone5上测试,拍照有有10几兆,最后encode 会丢失数据。
http://stackoverflow.com/questions/26152652/ios-html5-canvas-todataurl


0

huiyichengyi

赞同来自:

后台我用的是.net写的 我通过这种方式为什么拿不到上传的文件?我通过取Request集合中的Files文件集合拿的 拿到文件数量为0
HttpFileCollection MyFilecollection = Request.Files;
int cout = MyFilecollection.Count;
MyFilecollection[0].SaveAs(Server.MapPath("~/upload/" + MyFilecollection[0].FileName));

求解释


0

xunxun

赞同来自:

发现img.onload方法不执行,是什么原因啊


0

水粑粑

赞同来自:

以下是我编码成base64的字符串...不知道C#如何解码,求赐教




0

cj

赞同来自:

那个旋转90度的问题时怎么解决的呢?


0

cj

赞同来自:

您好,请问在三星上时如何兼容的呢?
input type=‘file’ 标签在应用中打不开。。。。请问您是如何解决的呢?


0

failedtocopy

赞同来自:

如果需要把截图获取到的base64数据生成成图片,保存到相册里面,这个要怎么实现呢,代码如下:
> view = Activity.getCurrentFocus();
plus.android.invoke(view,"setDrawingCacheEnabled",true);
plus.android.invoke(view,"buildDrawingCache");
//对view截图,存入bitmap中
bitmap = plus.android.invoke(view,"getDrawingCache");
if(null != bitmap){
console.log("截图成功");
try{
str = new StringBuffer();
bStream = new ByteArrayOutputStream();
bitmap.compress(bBitmap.CompressFormat.PNG, 100, bStream);
bStream.flush();
bStream.close();
bytes = bStream.toByteArray();
str.append(Base64.encodeToString(bytes, Base64.NO_WRAP));
} catch(e){
console.log(e.message);
}
var imgFile = "data:image/png;base64," + str.toString();
var filename = "order_" + id.toString() + ".jpg";

    plus.io.requestFileSystem(plus.io.PUBLIC_DOWNLOADS, function(fs) {
fs.root.getFile('test.png', {&quot;create&quot;: true},
function(fileEntry) {
console.log(fileEntry.fullPath);
fileEntry.createWriter(function(writer) {
writer.onwrite = function(e) {
console.log(&quot;Write data success!&quot;);
};
writer.write( str.toString() );
});
},function(e){
console.log(&quot;getFile failed: &quot; + e.message);
});
}, function(e) {
console.log(&quot;Request file system failed: &quot; + e.message);
});
\n

1.能生成一段base64编码 <img src=" ... "/> 这样也能生成图片,还能运行到console.log("Write data success!"); 这一句,但是到响应的目录下面找不到文件,怎么回事呢?
2.
2. 如果可以的话 能不能直接把在相册目录里面创建一个文件,然后把base64内容填进去,生成图片呢?


0

枫桥居APP

赞同来自:

有bug


0

_AJian_

赞同来自:

谢谢分享


0

DCloud_heavensoft

赞同来自:

HTML5+已经在plus.zip里提供了compressImage的原生图像压缩。
具体见:http://www.html5plus.org/doc/zh_cn/zip.html#plus.zip.compressImage


http://ask.dcloud.net.cn/article/123


0

huyong1978

赞同来自:

在iphone上压缩后,图片出现变形,高度没按等比生成,全部挤在顶部了,求解决


0

leejm

赞同来自:

php 服务器端怎么弄 多图上传的时候


0

妙妙

赞同来自:

请问怎么用mui.ajax提前图片到asp.net mvc api上呢,
我使用mui.post base64数据到asp.net mvc api上,后台根本收不到参数


0

roblade

赞同来自:

文件上传后再次下载下来能复原原来的照片质量吗?


0

SkyGrass

赞同来自:

先备注,我有预感,快用到了。


0

chrome

赞同来自:

请教一个问题:
包含这段代码的html页面,如果是从网上加载的,就会导致img.onload不执行,
而如果包含这段代码的html本身就在本地,就没有问题了。

http://ask.dcloud.net.cn/question/11337,这个是我提的问题,不知道表达清楚了么。


0

近水楼台

赞同来自:

mark


0

函数无机

赞同来自:

C# 服务端代码:

    [HttpPost]
public JsonResult Upload2() {
string dirPath = HttpContext.Server.MapPath(&quot;~/_Temp/&quot;);
if (!Directory.Exists(dirPath))
Directory.CreateDirectory(dirPath);
//System.IO.File.WriteAllText(dirPath + _FName + &quot;.txt&quot;, str);
MemoryStream ms = null;
try
{
byte[] byts = new byte[Request.InputStream.Length];
Request.InputStream.Read(byts, 0, byts.Length);
String strOrg = Server.UrlDecode(System.Text.Encoding.UTF8.GetString(byts));
String[] ss = strOrg.Split('&amp;');
foreach (String str in ss)
{
if (String.IsNullOrEmpty(str))
continue;

String _FName = new System.Random().Next(100000, 999999).ToString();
System.IO.File.WriteAllText(dirPath + _FName + &quot;.txt&quot;, str);
String _Ext = str.Substring(str.IndexOf(&quot;/&quot;) + 1, str.IndexOf(&quot;;base64,&quot;) - str.IndexOf(&quot;/&quot;) - 1);
String _Img = str.Substring(str.IndexOf(&quot;;base64,&quot;) + 8);
byte[] arr = Convert.FromBase64String(_Img);
ms = new MemoryStream(arr);
System.Drawing.Bitmap bmp = new System.Drawing.Bitmap(ms);
switch (_Ext.ToUpper())
{
case &quot;PNG&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Png);
break;
case &quot;JPG&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Jpeg);
break;
case &quot;JPEG&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Jpeg);
break;
case &quot;BMP&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Bmp);
break;
case &quot;GIF&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Gif);
break;
case &quot;ICON&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Icon);
break;
case &quot;TIFF&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Tiff);
break;
case &quot;WMF&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Wmf);
break;
case &quot;EMF&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Emf);
break;
case &quot;EXIF&quot;:
bmp.Save(dirPath + _FName + &quot;.&quot; + _Ext, System.Drawing.Imaging.ImageFormat.Exif);
break;
}
ms.Close();
ms.Dispose();
ms = null;
}

return Json(new
{
success = true,
message = &quot;上传成功.&quot;
}, JsonRequestBehavior.AllowGet);
}
catch (Exception ex)
{
return Json(new
{
success = false,
message = Helper.GetExceptionFullMessage(ex)
}, JsonRequestBehavior.AllowGet);
}
finally {
if (ms != null)
{
ms.Close();
ms.Dispose();
ms = null;
}
}
}
\n
\n
        function imgupgrade() {
plus.nativeUI.showWaiting();
var url = 'http://192.168.1.112/hb/Fm/Archives/Upload2';
var dataType = 'json';
//发送数据
var data = {
files1: f1 //base64数据
};
mui.post(url, data, success, dataType);
}
//成功响应的回调函数
var success = function(response) {
plus.nativeUI.closeWaiting();
if (response != null &amp;&amp; !response.success) {
alert(response.message);
}
}
\n
0

东哥的苹果

赞同来自:

一般很友好的做法是从相册选取图片,然后裁剪,最后上传(京东,淘宝,支付宝等混合开发出来的app都这么做),感觉压缩图片不好,那么问一下,选取图片后裁剪怎么做?


0

周羊羊

赞同来自:

拍照之后APP会重启 怎么办?


0

墓志家族

赞同来自:

怎么用不了


0

317560315@qq.com

赞同来自:

先留名、以后会用到


0

明明航 - 你好

赞同来自:

if(files.length < 2) 这一步报错(拍照上传)
Can't find variable: files


0

GuanQun - 还没想好

赞同来自:

谢谢谢谢


0

ronger

赞同来自:

学习一下


0

rabet - 开发,技术服务

赞同来自:

多张图压缩 处理方法
解决plus.zip.compressImage 图片循环压缩无法都压缩完的问题
解决多图压缩处理方法


0

1019422167@qq.com

赞同来自:

mark


0

1019422167@qq.com

赞同来自:

mark


0

18768156@qq.com

赞同来自:

index


0

1466744324@qq.com - nacissus

赞同来自:

php端应该这样
<?php
namespace Home\Controller;
use Think\Controller;
//登录控制器
class TestController extends Controller {
//显示登录页面并验证
public function index(){

    $s=dirname(__FILE__); //获的服务器路劲
$time =time(); //获得当前时间戳
$files =$_POST['files1'];
$key = str_replace(' ','+',$files);
$files1 = substr($key,23); //百度一下就可以知道base64前面一段需要清除掉才能用。
//解码
$tmp = base64_decode($files1);
//$fp=$s.&quot;/uploads/&quot;.$time.&quot;.jpg&quot;; //确定图片文件位置及名称
//写文件
// file_put_contents( $fp, $tmp); //给图片文件写入数据
\n

file_put_contents("./mylog.jpg",$tmp);
echo 1;
}

}


0

262365913@qq.com

赞同来自:

图片旋转90度的没找到解决方案。。。。


0

llyzlc

赞同来自:

压缩后上传的是base64,怎样不上传base64,而是上传的file


要回复问题请先登录注册

退出全屏模式 全屏模式 回复