Изменение размера изображения на стороне клиента с помощью JavaScript перед загрузкой на сервер - PullRequest
141 голосов
/ 12 марта 2010

Я ищу способ изменения размера изображения на стороне клиента с помощью JavaScript (действительно изменение размера, а не только изменение ширины и высоты).
Я знаю, что это возможно сделать во Flash, но я хотел бы избежать этого, если это возможно.

Есть ли какой-нибудь алгоритм с открытым исходным кодом где-нибудь в сети?

Ответы [ 7 ]

99 голосов
/ 28 июля 2015

Вот суть, которая делает это: https://gist.github.com/dcollien/312bce1270a5f511bf4a

(версия es6 и версия .js, которые могут быть включены в тег скрипта)

Вы можете использовать его следующим образом:

<input type="file" id="select">
<img id="preview">
<script>
document.getElementById('select').onchange = function(evt) {
    ImageTools.resize(this.files[0], {
        width: 320, // maximum width
        height: 240 // maximum height
    }, function(blob, didItResize) {
        // didItResize will be true if it managed to resize it, otherwise false (and will return the original file as 'blob')
        document.getElementById('preview').src = window.URL.createObjectURL(blob);
        // you can also now upload this blob using an XHR.
    });
};
</script>

Он включает в себя множество средств обнаружения поддержки и полифилов, чтобы обеспечить работу на максимально возможном количестве браузеров.

(он также игнорирует GIF-изображения - в случае, если они анимированы)

58 голосов
/ 21 апреля 2011

Ответ на этот вопрос - да - в HTML 5 вы можете изменять размеры изображений на стороне клиента, используя элемент canvas. Вы также можете взять новые данные и отправить их на сервер. Смотрите этот урок:

http://hacks.mozilla.org/2011/01/how-to-develop-a-html5-image-uploader/

12 голосов
/ 23 августа 2011

Если вы изменяли размер перед загрузкой, я только что узнал это http://www.plupload.com/

Он делает всю магию для вас любым мыслимым способом.

К сожалению, только изменение размера HTML5 поддерживается браузером Mozilla, но вы можете перенаправить другие браузеры на Flash и Silverlight.

Я только что попробовал, и он работал с моим Android!

Я использовал http://swfupload.org/ во флэш-памяти, он отлично справляется с работой, но размер изменения очень мал. (не может вспомнить предел) и не возвращается к html4, когда флэш-память недоступна.

10 голосов
/ 21 апреля 2016

http://nodeca.github.io/pica/demo/

В современном браузере вы можете использовать canvas для загрузки / сохранения данных изображения. Но вы должны помнить несколько вещей, если вы изменяете размер изображения на клиенте:

  1. У вас будет только 8 бит на канал (jpeg может иметь лучший динамический диапазон, около 12 бит). Если вы не загружаете профессиональные фотографии, это не должно быть проблемой.
  2. Будьте осторожны с алгоритмом изменения размера. Большинство средств изменения размера на стороне клиента используют тривиальную математику, и результат хуже, чем мог бы быть.
  3. Возможно, вам потребуется повысить резкость уменьшенного изображения.
  4. Если вы хотите повторно использовать метаданные (exif и другие) из оригинала - не забудьте удалить информацию из цветового профиля. Потому что он применяется при загрузке изображения на холст.
6 голосов
/ 12 марта 2010

Возможно, с тэгом canvas (хотя он не переносимый). Есть блог о том, как повернуть изображение с помощью canvas здесь , я думаю, если вы можете повернуть его, вы можете изменить его размер. Может быть, это может быть отправной точкой.

См. там также.

3 голосов
/ 28 мая 2018

Вы можете использовать инфраструктуру обработки изображений javascript для обработки изображений на стороне клиента перед загрузкой изображения на сервер.

Ниже я использовал MarvinJ для создания исполняемого кода на основе примера на следующей странице: «Обработка изображений на стороне клиента перед его загрузкой на сервер»

В основном я использую метод Marvin.scale (...) для изменения размера изображения. Затем я загружаю изображение в виде большого двоичного объекта (используя метод image.toBlob () ). Сервер отвечает обратно, предоставляя URL полученного изображения.

/***********************************************
 * GLOBAL VARS
 **********************************************/
var image = new MarvinImage();

/***********************************************
 * FILE CHOOSER AND UPLOAD
 **********************************************/
 $('#fileUpload').change(function (event) {
	form = new FormData();
	form.append('name', event.target.files[0].name);
	
	reader = new FileReader();
	reader.readAsDataURL(event.target.files[0]);
	
	reader.onload = function(){
		image.load(reader.result, imageLoaded);
	};
	
});

function resizeAndSendToServer(){
  $("#divServerResponse").html("uploading...");
	$.ajax({
		method: 'POST',
		url: 'https://www.marvinj.org/backoffice/imageUpload.php',
		data: form,
		enctype: 'multipart/form-data',
		contentType: false,
		processData: false,
		
	   
		success: function (resp) {
       $("#divServerResponse").html("SERVER RESPONSE (NEW IMAGE):<br/><img src='"+resp+"' style='max-width:400px'></img>");
		},
		error: function (data) {
			console.log("error:"+error);
			console.log(data);
		},
		
	});
};

/***********************************************
 * IMAGE MANIPULATION
 **********************************************/
function imageLoaded(){
  Marvin.scale(image.clone(), image, 120);
  form.append("blob", image.toBlob());
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<script src="https://www.marvinj.org/releases/marvinj-0.8.js"></script>
<form id="form" action='/backoffice/imageUpload.php' style='margin:auto;' method='post' enctype='multipart/form-data'>
				<input type='file' id='fileUpload' class='upload' name='userfile'/>
</form><br/>
<button type="button" onclick="resizeAndSendToServer()">Resize and Send to Server</button><br/><br/>
<div id="divServerResponse">
</div>
2 голосов
/ 12 июня 2016

Да, с современными браузерами это вполне выполнимо. Даже выполнимо до такой степени, что вы загружаете файл специально как двоичный файл, выполнив любое количество изменений холста.

http://jsfiddle.net/bo40drmv/

(этот ответ является улучшением принятого ответа здесь )

Помните, что процесс обработки результатов в PHP должен быть похож на:

//File destination
$destination = "/folder/cropped_image.png";
//Get uploaded image file it's temporary name
$image_tmp_name = $_FILES["cropped_image"]["tmp_name"][0];
//Move temporary file to final destination
move_uploaded_file($image_tmp_name, $destination);

Если кто-то беспокоится о точке зрения Виталия, вы можете попробовать немного обрезать и изменить размер рабочего jfiddle.

...