Обрезка с использованием Paperclip, ImageMagick, Jcrop и S3: почему бы не использовать image.reprocess!переработать? - PullRequest
6 голосов
/ 07 июля 2011

Я (слабо) следую RailsCasts, учебному пособию # 182 , в котором используются Paperclip, ImageMagick и Jcrop, чтобы разрешить произвольное обрезание загруженных изображений.

Поскольку я использую Amazon S3 для хранения файлов, мне пришлось перенастроить части учебника в соответствии с требованиями. Кажется, что все работает идеально , за исключением , поскольку тот факт, что обрезанная версия моего изображения не обрабатывается повторно (или результат этой повторной обработки не перезагружается на S3), - после процесса обрезки У меня осталось то же изображение, которое я изначально загрузил (это верно для всех размеров изображений, которые я храню для каждого изображения).

Вот моя особенность (как на картинке):

class Feature < ActiveRecord::Base
  require "#{Rails.root}/lib/paperclip_processors/cropper.rb"

  attr_accessible    :image_file_name, :image
  attr_accessor      :crop_x, :crop_y, :crop_w, :crop_h
  after_update       :reprocess_image, :if => :cropping?

  if Rails.env == "production"
  S3_CREDENTIALS = { :access_key_id     => '<REDACTED>',
                     :secret_access_key => '<REDACTED>',
                     :bucket            => "<REDACTED>"}
  else
  S3_CREDENTIALS = { :access_key_id     => '<REDACTED>',
                     :secret_access_key => '<REDACTED>',
                     :bucket            => "<REDACTED>"}
  end

  has_attached_file :image,
                    :styles          => { :small => "240x135>", :croppable => "960x960>", :display => "960x540>" },
                    :processors      => [:cropper],
                    :storage         => :s3,
                    :s3_credentials  => S3_CREDENTIALS,
                    :path            => "features/:id/:style.:extension"

  validates_attachment_content_type :image, :content_type => ['image/jpeg', 'image/gif', 'image/png',
                                                              'image/pjpeg', 'image/x-png'], 
                                            :message => 'must be a JPEG, GIF or PNG image'

  def cropping?
    !crop_x.blank? && !crop_y.blank? && !crop_w.blank? && !crop_h.blank?
  end

  def image_geometry(style = :original)
    @geometry ||= {}
    path = (image.options[:storage]==:s3) ? image.url(style) : image.path(style)
    @geometry[style] ||= Paperclip::Geometry.from_file(path)
  end

  private

  def reprocess_image
    image.reprocess!
  end
end

Вот мой 'cropper.rb' (процессор Paperclip):

module Paperclip
  class Cropper < Thumbnail
    def transformation_command
      if crop_command
        crop_command + super.sub(/ -crop \S+/, '')
      else
        super
      end
    end

    def crop_command
      target = @attachment.instance
      if target.cropping?
        " -crop '#{target.crop_w}x#{target.crop_h}+#{target.crop_x}+#{target.crop_y}'"
      end
    end
  end
end

Соответствующие действия моего FeatureController:

class FeaturesController < ApplicationController

  def new
    @feature = Feature.new
  end

  def create
    @feature = Feature.new(params[:feature])
    if @feature.save
      if params[:feature][:image].blank?
        flash[:notice] = "New feature added!"
        redirect_to @feature
      else
        render :crop
      end
    else
      @title = "Add a New Feature"
      render :new
    end
  end

  def edit
    @feature = Feature.find(params[:id])
    @title = "Edit #{@feature.headline}"
  end

  def update
    @feature = Feature.find(params[:id])
    if @feature.update_attributes(params[:feature])
      if params[:feature][:image].blank?
        flash[:notice] = "Feature updated!"
        redirect_to @feature
      else
        render :crop
      end
    else
      @title = "Edit Feature"
      render :edit
    end
  end
end

И соответствующие строки моего представления 'crop.html.erb':

<% content_for :javascript_includes do %>
    <%= javascript_include_tag 'jquery.Jcrop.min' %>
    <script type="text/javascript" charset="utf-8">
    $(function() {
        $('#cropbox').Jcrop({
            onChange: update_crop,
            onSelect: update_crop,
            setSelect: [0, 0, 960, 540],
            aspectRatio: 960/540
        });
    });

    function update_crop(coords) {
        var ratio = <%= @feature.image_geometry(:original).width %> / <%= @feature.image_geometry(:croppable).width %>;
        $("#crop_x").val(Math.round(coords.x * ratio));
        $("#crop_y").val(Math.round(coords.y * ratio));
        $("#crop_w").val(Math.round(coords.w * ratio));
        $("#crop_h").val(Math.round(coords.h * ratio));
    };
    </script>
<% end %>
<% content_for :style_includes do %>
    <%= stylesheet_link_tag 'jquery.Jcrop', :media => 'screen' %>
<% end %>

<%= image_tag @feature.image.url(:croppable), :id => "cropbox" %>

<% form_for @feature do |f| %>
    <% for attribute in [:crop_x, :crop_y, :crop_w, :crop_h] %>
        <%= f.hidden_field attribute, :id => attribute %>
    <% end %>
    <p><%= f.submit "Crop" %></p>
<% end %>

Проблема не в том, что есть ошибка с пользовательской обрезкой (смещение, площадь обрезки и т. Д.), А в том, что обрезки не происходит , когда я нажимаю «обрезать» - Я только что оставил изображения, полученные с оригинальной загрузки / процесса. Похоже, что image.reprocess! происходит вообще (или результаты повторной обработки не сохраняются в S3).

Почему это может быть, и что я могу с этим поделать?

Ответы [ 3 ]

3 голосов
/ 01 марта 2012

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

ошибка скрепки при повторной обработке после обновления рельсов 3

2 голосов
/ 05 сентября 2011

Хорошо, позвольте мне попытаться помочь:)

Во-первых, не можете ли вы включить в модель свой процессор Paperclip, пусть Paperclip справится с этим.

Во-вторых, удалите :after_update и замените на :before_update, который должен установить image.options[ :crop ] для процессора. В вашем процессоре попробуйте это:

def initialize file, options = {}, attachment = nil
  super
  #......

  @crop = options[ :crop ]

  #......

В-третьих, измените ваш transformation_command в процессоре:

def transformation_command
  if @crop
      trans = " -quality 75"
      trans << " -crop \"#{<YOUR_CODE>}\" +repage" if @crop
      trans
  end
end

А потом опубликуйте свои выводы:)

0 голосов
/ 18 сентября 2011

Я также использовал то же видео с RailsCast, чтобы помочь мне начать обрезку.У меня была похожая проблема с ошибкой NOSUCHKEY, которую я смог отследить до повторной обработки!вызов, который был вызван after_update.Я смог решить проблему, используя ответ Виша и изменив его.Возможно, мое решение не совсем то, что вам нужно, поскольку я обрезаю каждый раз.

Я просто удалил вызов after_update для повторной обработки!и оставил все остальное на месте.Это привело к тому, что переданный в процессор (в вашем случае кроппер) использовался вместо значения по умолчанию и происходит перед загрузкой изображения на S3.Поскольку вы обрезаете файл перед его первой загрузкой, все работает!

Если вы хотите использовать условную обрезку, то вам нужно сделать то, что предложил Виш, - установить переменную или передать параметр с изображениемобъект в before_update.Он должен быть доступен в вашем пользовательском процессоре, чтобы вы могли разместить свою логику условного кадрирования на основе ее значения.

...