Отправка канвы на Rails-бекенд

rails4 + paperclip + canvas

В интернетах всё ещё встречаются сайты, где от пользователя требуются сложные манипуляции с изображениями в графических редакторах просто для того, чтобы поставить котика на аву. Однако с помощью HTML5-элемента canvas можно не только самостоятельно отмасштабировать и обрезать котика, но и, например, наложить на него приятный глитч-эффект. И всё это на стороне клиента: без отправки котиков на бекенд и обратно.

Почему же мир так несправедлив?

Проблематичным моментом в этом процессе является нетривиальность отправки и получения канвы, так как canvas не является input-элементом и поэтому не передаётся при отправке формы, а также не может быть получен как обычный файл на стороне бекенда.

Однако содержимое канвы можно стандартным образом преобразовывать в base64-строки вида



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

Фронтенд

Так как канва не может быть непосредственно передана как файл, то нам понадобится дополнительное input-поле для передачи её base64-репрезентации. Так что лучше бы вам удалить поля file_field, если они у вас ещё есть, и добавить новое скрытое поле для передачи данных канвы

<%= f.hidden_field :canvas_file %>

а также соответствующий onDomReady-обработчик, предворяющий отправку формы кодом, который свяжет это поле с содержимым канвы:

$(function(){
  $('#<id_of_your_form>').submit(function() {
    var canvas = $('#canvas')[0]
    $('#canvas_file').val(canvas.toDataURL());
  })
})

Ключевым здесь является метод toDataURL, собственно, превращающий содержимое канвы в base64-репрезентацию png-изображения, соответствующего содержимому канвы.

Кстати, через toDataURL можно декодиться и в jpeg с указанием качества, только нужно понимать, что это будет пересжатие с потерями. Ещё, не знаю, с чем это связанно, но toDataURL из консоли Chrome у меня по крайней мере возвращает изображение типа "no resource" вместо актуального содержимого.

Декодер

Расшифровывать получаемую base64-строку можно с помощью модуля datafy.rb. Для чего достаточно расположить его в lib и загружать перед использованием.

Для автозагрузки модулей из lib, можно добавить в config/application.rb строку

config.autoload_paths << Rails.root.join('lib')

либо загружать модуль вручную.

Кажется, он также существует в виде гема.

Бекенд

На бекенде нужно декодировать base64-строку и передать её в paperclip (или другой обработчик загрузок).

Учитывая, что мы добавили новое поле в форму, мы обновляем нашу модель, добавляя атрибут canvas_file (и не забывая разрешить его в контроллере), проверяем, что тип png — валидный mime-тип для загрузчика (иначе загрузка канвы будет невозможна), выполняем декодирование и передаём результат в загрузчик через временный файл:

attr_accessor :canvas_file

has_attached_file :file
validates_attachment :file, content_type: { content_type: ["image/jpg", "image/jpeg", 'image/png'] }

before_validation :save_canvas_file

private
  def save_canvas_file
    f = File.open("tmp/reply.png", "wb")
    data, mediatype = Datafy::decode_data_uri(canvas_file)
    f.write(data)
    f.close
    self.file = File.open("tmp/reply.png", "r")
  end

Изображение временно сохраняется в проектной папке tmp:

$ file tmp/reply.png
reply.png: PNG image data, 240 x 240, 8-bit/color RGBA, non-interlaced

 

 

Статья основана на исходниках git-пользователя pierrevalade и утилите datafy.rb оттуда же (автор модуля — Eric Hodel).

 

 

 

shitpoet@gmail.com

 



 

free hit counters