Вступление
Всем привет! Здороваюсь с хабром я в первый, и надеюсь не последний, раз. Не смотря на то, что читаю хабр довольно давно, идея написать что-то полезное появилась совсем недавно, когда на работе я столкнулся с весьма интересной задачей — разработка он-лайн редактора коллажей. Поскольку особого ассортимента инструментов разработки не было, решили делать средствами js+jQuery и php GD. Процесс реализации задуманного оказался весьма интересным, и куча полученных положительных эмоций и новых навыков подтолкнули меня на написание статьи на хабр. В этой статейке я постараюсь рассказать о некоторых интересных моментах, с которыми столкнулся при разработке он-лайн редактора.
Задача
По изначальному плану статьи я хотел описать весь процесс разработки, но потом передумал, поскольку статья получилась бы слишком длинной и имела бы много очевидных и итак всем понятных вещей. Поэтому план статьи был переработан, и я решил оставить только самые интересные и важные, как мне кажется, моменты.
Итого: речь пойдет об использовании jQuery UI в связке с PHP библиотекой GD. В статье я постараюсь, как можно доходчивее, показать и рассказать об использовании таких возможностей jQuery UI, как перетаскивание и ресайз элементов. А также формирование картинки из созданных и обработанных пользователем элементов (картинок).
Чтобы было более понятней и наглядней думаю будет не плохо сделать рабочий пример(посмотреть можно тут). В примере реализована одна из частей он-лайн редактора, а именно работа с аппликациями, в которой пользователь может наложить на картинку дополнительные элементы, перетаскивать их как угодно и ресайзить, после чего все это «искусство» должно собраться в единую картинку.
Что-то я много говорю, пора уже и к делу приступить, начнем.
Решение
Для нетерпеливых сразу выложу рабочий пример: ссылка
И исходники: ссылка
Для начала стоит определиться, что мы будем использовать и где.
На стороне клиента (в браузере), будем пользоваться, не нуждающийся в представлении, библиотекой jQuery и несколькими ее плагинами, а именно Draggable и Resizable.
На сервере будем использовать php и библиотеку GD, которая установлена практически на каждом сервере и хостинге, в отличии от более продвинутого аналога ImageMagick.
Процесс разработки начнем с клиентской части. Нам потребуется создать рабочую область, в которой пользователь сможет таскать и ресайзить картинки. А также необходимо добавить на страницу панель, в которой будет находится несколько вариантов аппликаций.
Покажу небольшой кусочек html, для лучшего понимания:
<div class="work_area">
<img src="/resources/images/angelina.jpg" width="500" height="600" id="main_img_big" />
</div>
<div class="applications_div">
<img src="/resources/applications/1.png" width="64" height="64" />
<img src="/resources/applications/2.png" width="64" height="64" />
<img src="/resources/applications/3.png" width="64" height="64" />
<img src="/resources/applications/4.png" width="64" height="64" />
<img src="/resources/applications/5.png" width="64" height="64" />
<img src="/resources/applications/6.png" width="64" height="64" />
<img src="/resources/applications/7.png" width="64" height="64" />
</div>
Нам необходимо добавлять элементы аппликации на рабочую область, когда пользователь кликнет по приглянувшейся картинке-аппликации. Поскольку количество добавленных элементов ни как нельзя предугадать, добавление нужно сделать динамическим, для этого можно использовать следующий код:
var num_elem = 0;
// добавление аппликации в рабочую область
function addApplication(element){
var applicImg = element.clone(); // создаем копию элемента
// удаляем аттрибуты размеров картинки
applicImg.removeAttr('width');
applicImg.removeAttr('height');
// добавляем родительский див для аппликации в рабочую область
var allElement = '<div class="applic_new_el_div" id="move_applic_'+num_elem+'"><span class="close_applic"></span></div>';
$('.work_area').append(allElement);
// добавляем класс для перетаскивания
applicImg.addClass('applic_new_el');
// задаем место появления в рабочей области
$('#move_applic_'+num_elem).css({
'top': '0px',
'left': '0px'
});
applicImg.attr('id', 'applic_'+num_elem);
// добавляем элемент
$('#move_applic_'+num_elem).append(applicImg);
init_drag(num_elem); // задаем перетаскивание
init_resize(num_elem); // задаем резайз
num_elem ++; // увиличение счетчика для аппликаций
}
Как вы уже обратили внимание, добавляется не только картинка, но еще и вместе с дивом, в который обернута и span`ом. Span будет выполнять роль «крестика», который будет при необходимости удалять не нужную аппликацию из рабочей области.
После добавления аппликации, необходимо вызвать плагины, для добавления возможности ресайза и перемещения картиночки.
Для возможности ресайза будем использовать плагин Resizable, вызовем его с помощью такой функции:
// ресайз для аппликаций
function init_resize(num_el){
$('#move_applic_'+num_el).resizable({
aspectRatio: true, // сохранять пропорции
handles: 'ne, nw, se, sw', // имена классов для угловых блоков
alsoResize: "#applic_"+num_el // расайзим еще и родительский див - рамку
});
}
Чтобы картинка еще и перемещалась по рабочей области, воспользуемся плагином Draggable, для его вызова напишем вот такую функцию:
// задаем перетаскивание для апликации
function init_drag(num_el){
$('#move_applic_'+num_el).draggable({
cursor: 'move', // вид курсора
containment: '.work_area', // ограничение перемещения
scroll: false, // автоскроллинг
drag: null // событие при перемещении
});
}
На этом работа с аппликацией закончена. Все что остается сделать на клиентской стороне – это собрать все данные о добавленных аппликациях и отправить на сервер для обработки. Чтобы создать на сервере такую же картинку, как и в браузере, нам потребуется знать путь до каждой картинки-аппликации, размер (ширину и высоту) и положение на странице, относительно рабочей области.
Чтобы не делать перезагрузку страницы, а также для удобства передачи данных, воспользуемся ajax`ом. Код для сбора данных об аппликациях и отправки не сервер выглядит вот так:
// создание картинки с наложением аппликации. Запрос на сервер
function ajaxMakeImage(){
// объявляем необходимые массивы
var arrayWidth = [];
var arrayHeight = [];
var arraySrc = [];
var arrayTop = [];
var arrayLeft = [];
var srcImage = $('#main_img_big').attr('src');
var workAreaTop = $('.work_area').offset().top;
var workAreaLeft = $('.work_area').offset().left;
var num = 0;
$('.applic_new_el_div').each(function(e) {
arrayWidth[num] = $(this).width();
arrayHeight[num] = $(this).height();
arraySrc[num] = $(this).children('.applic_new_el').attr('src');
arrayTop[num] = $(this).offset().top;
arrayLeft[num] = $(this).offset().left;
num++;
});
// отправляем данные на сервер
$.ajax({
type: "POST",
url: "/ajax_action.php",
data: {
'arraySrc': arraySrc, // массив путей для аппликаций
'arrayWidth': arrayWidth, // массив длин аппликаций
'arrayHeight': arrayHeight,// массив ширины аппликаций
'arrayTop': arrayTop, // массив отступов сверху для аппликаций
'arrayLeft': arrayLeft, // массив отступов слева для аппликаций
'srcImage': srcImage, // ссылка на фотографию(главная картинка)
'workAreaTop': workAreaTop, // отступ сверху до рабочей области
'workAreaLeft': workAreaLeft, // отступ слева до рабочей области
},
dataType: "json",
success: function(data){
if(data.result == 'success'){
// если все прошло успешно
// выводим готовую картинку
$('#test_show').attr('src', data.imgSrc);
alert('Картинка создана');
}else{
// error
// @todo вывод ошибки
}
}
});
}
Работа в браузере закончена. Теперь необходимо написать на сервере скрипт, который будет обрабатывать полученные данные и опираясь на них генерировать картинку.
Как я уже говорил, для работы с картинками будем использовать библиотеку GD.
Поскольку аппликации могут быть абсолютно любого размера, то нужно каждую полученную аппликацию нужно ресайзить. Напишем для этого небольшую функцию:
function resizePhotoPNG($source, $path, $height, $width){
$rgb = 0xffffff; //цвет заливки фона
$size = getimagesize($source);//узнаем размеры исходной картинки
$xRatio = $width / $size[0]; //пропорция ширины
$yRatio = $height / $size[1]; //пропорция высоты
$ratio = min($xRatio, $yRatio);
$kRatio = ($xRatio == $ratio); //соотношения ширины к высоте
$new_width = $kRatio ? $width : floor($size[0] * $ratio); //ширина
$new_height = !$kRatio ? $height : floor($size[1] * $ratio); //высота
// расхождение с заданными параметрами по ширине
$new_left = $kRatio ? 0 : floor(($width - $new_width) / 2);
// расхождение с заданными параметрами по высоте
$newTop = !$kRatio ? 0 : floor(($height - $new_height) / 2);
//создаем вспомогательное изображение пропорциональное картинке
$img = imagecreatetruecolor($width, $height);
imagealphablending($img, false);
imagesavealpha($img, true);
$photo = imagecreatefrompng($source); //достаем наш исходник
imagecopyresampled($img, $photo, $new_left, $newTop, 0, 0, $new_width, $new_height, $size[0], $size[1]); //копируем на него превью с учетом расхождений
imagepng($img, $path); //сохраняем результат
// Очищаем память после выполнения скрипта
imagedestroy($img);
imagedestroy($photo);
// вернем путь для картинки
return $path;
}
Теперь, имея функцию ресайза, остается только обработать все аппликации с ее помощью и сохранить во временной папке. После этого необходимо на главную картинку по очереди нанести каждую аппликацию, конечно не забывая их смещать по осям X и Y. Смещения по осям будут равны отступам слева и сверху в браузере, относительно рабочей области.
Нанесение со смещением можно сделать следующим образом:
// $arrayApplication – массив путей до уже отресайзеных аппликаций
// $mainImg – это главная рабочая картинка
foreach($arrayApplication as $k=>$oneAppl){
//Загружаем одну аппликацию и задаем прозрачность
$imageFon = imagecreatefrompng($oneAppl);
imagealphablending($imageFon, false);
imagesavealpha($imageFon, true);
// совмещаем картинки
imagecopy($mainImg, $imageFon, $applX[$k]-$imgX, $applY[$k]-$imgY, 0, 0, imagesx($imageFon), imagesy($imageFon));
}
На этом обработка картинок закончена, остается только сохранить получившуюся картинку и передать путь до нее в браузер.
Для сохранения используем следующий код:
imageJpeg($mainImg, $pathForImg, 100); // сохранение картинки в папку $pathForImg и с качеством 100
// и не забудем очистить память
imagedestroy($mainImg);
Теперь отправляем в браузер ответ, в котором передадим путь до созданной картинки:
$result = array(
'result' => 'success', // результат работы скрипта
'imgSrc' => $resultSrc // путь до картинке
);
echo json_encode($result); // кодируем массив в JSON и передаем данные браузеру
На этом наложение аппликаций на картинку закончено.
Если меня совсем уж не заминусуют, и статья окажется хоть кому-то полезной, я постараюсь написать еще несколько статей по созданию коллажа (загрузка фотографий, наложение фильтров, наложение текста и тд).
Список полезных ссылок, опираясь на которые я писал статью:
– Обработка изображений и GD
– Плагин jQuery Draggable
– Плагин jQuery Resizable
– Рабочий пример
– Исходники примера
Спасибо за внимание!