OSDN Git Service

t#30328:add export
[pettanr/pettanr.git] / app / models / picture.rb
index e5bbda4..a64ce6e 100644 (file)
@@ -11,75 +11,120 @@ class Picture < ActiveRecord::Base
   validates :width, :presence => true, :numericality => true, :natural_number => true
   validates :height, :presence => true, :numericality => true, :natural_number => true
   validates :filesize, :presence => true, :numericality => {:greater_than => 0, :less_than_or_equal_to => 2000000}, :natural_number => true
+  validates :md5, :presence => true, :length => {:minimum => 32, :maximum => 32}
   validates :license_id, :presence => true, :numericality => true, :existence => true
   validates :artist_id, :presence => true, :numericality => true, :existence => true
   validates :artist_name, :presence => true
   validates :classname, :presence => true, :length => {:maximum => 50}
   
-  def dext
-    self.ext.downcase
+  def supply_default
+  end
+  
+  def overwrite rp
+    attr = {:width => rp.width, :height => rp.height, :ext => rp.ext, :filesize => rp.filesize, 
+      :original_picture_id => rp.original_picture_id, :license_id => rp.license_id, :artist_id => rp.artist_id, 
+      :md5 => rp.md5, :artist_name => rp.artist_name, :classname => rp.classname, :credit => rp.credit, :settings => rp.settings
+    }
+    self.attributes = attr
+    self.revision = self.new_revision   #Do not move to attr. new_revision reffernces self.original_picture_id
+  end
+  
+  def own? ar
+    if ar.is_a?(Author)
+      self.artist_id == ar.artist.id
+    elsif ar.is_a?(Artist)
+      self.artist_id == ar.id
+    else
+      false
+    end
+  end
+  
+  def visible? ar
+    return true
+  end
+  
+  def showable? ar
+    return false unless ar.is_a?(Artist)
+    return true if self.own?(ar)
+    self.enable? and self.head?
   end
   
   def filename
-    "#{self.id}.#{self.dext}"
+    "#{self.id}.#{self.ext}"
+  end
+  
+  def gifname
+    "#{self.id}.gif"
   end
   
   def mime_type
-    "image/#{self.dext}"
+    "image/#{self.ext}"
+  end
+  
+  def url
+    '/pictures/' + filename
+  end
+  
+  def opt_img_tag
+    {:src => self.url, :width => self.width, :height => self.height}
+  end
+  
+  def tmb_opt_img_tag
+    tw, th = PettanImager.thumbnail_size(self.width, self.height)
+    {:src => self.url, :width => tw, :height => th}
   end
   
   def new_revision
     Picture.maximum(:revision, :conditions => ['original_picture_id = ?', self.original_picture_id]).to_i + 1
   end
   
-  def v(rimg)
-    rimg.flip.to_blob
+  def enable?
+    r = self.head.resource_picture
+    r ? true : false
   end
   
-  def h(rimg)
-    rimg.flop.to_blob
+  def head
+    Picture.find(:first, :conditions => ['original_picture_id = ?', self.original_picture_id], :order => 'pictures.revision desc')
   end
   
-  def vh(rimg)
-    rimg.flip.flop.to_blob
+  def head?
+    self == head
   end
   
   def to_gif?
-    self.dext == 'png' and self.flag_gif_convert >= 0
-  end
-  
-  def self.png_to_gif(data)
-    res = nil
-    begin
-      mgk = Magick::Image.from_blob(data).shift
-      mgk.format = 'gif'
-      res = mgk
-    rescue
-      res = false
-    end
-    res
+    self.ext == 'png' and self.flag_gif_convert >= 0
   end
   
   def subdirs
     flag_reverse < 0 ? [''] : ['', 'v', 'h', 'vh']
   end
   
-  def copy_data(rp)
-    attr = {:width => rp.width, :height => rp.height, :ext => rp.ext, :filesize => rp.filesize, 
-      :original_picture_id => rp.original_picture_id, :license_id => rp.license_id, :artist_id => rp.artist_id, 
-      :artist_name => rp.artist_name, :classname => rp.classname, :credit => rp.credit, :settings => rp.settings
-    }
-    self.attributes = attr
+  def self.find_by_md5 md5
+    r = Picture.find :all, :conditions => ['pictures.md5 = ?', md5], :order => 'pictures.updated_at desc'
   end
   
-  def store(mgk)
+  def self.list_by_md5 md5, opid
+    r = Picture.find :all, :conditions => ['pictures.md5 = ? and pictures.original_picture_id <> ?', md5, opid], :order => 'pictures.updated_at desc'
+  end
+  
+  def self.exist_by_md5 md5, opid
+    Picture.list_by_md5(md5, opid).empty? ? false : true
+  end
+  
+  def self.show rid, au
+    opt = {}
+    r = Picture.find(rid, opt)
+    raise ActiveRecord::Forbidden unless r.visible?(au)
+    r
+  end
+  
+  def store(imager)
     res = false
-    self.revision = self.new_revision
     if res = self.save
-      if res = self.store_picture(mgk)
+      if res = self.store_picture(imager, self.filename)
         if self.to_gif?
-          if gifmgk = Picture.png_to_gif(mgk.to_blob)
-            res = self.store_picture(gifmgk)
+          if gifimager = imager.to_gif
+            res = self.store_picture(gifimager, self.gifname)
           else
             res = false
           end
@@ -89,28 +134,50 @@ class Picture < ActiveRecord::Base
     res
   end
   
-  def store_picture(mgk)
+  def store_picture(imager, fn)
     res = false
     subdirs.each do |d|
-      picdata = d.empty? ? mgk.to_blob : self.__send__(d, mgk)
-      res = PictureIO.picture_io.put(picdata, "#{self.id}.#{mgk.format}", d)
+      picdata = d.empty? ? imager.binary : imager.__send__(d)
+      res = PictureIO.picture_io.put(picdata, fn, d)
       break unless res
     end
     res
   end
   
   def restore(subdir = nil)
-    PictureIO.resource_picture_io.get self.filename, subdir
+    PictureIO.picture_io.get self.filename, subdir
   end
   
-  def self.show pid
-    r = Picture.find(pid)
-#    raise ActiveRecord::Forbidden unless c.visible?(au)
-    r
+  def self.export(dt = nil)
+    opt = {}
+    cond = if dt
+      ['artists.author_id is not null and pictures.updated_at >= ?', dt]
+    else
+      'artists.author_id is not null'
+    end
+    opt.merge!({:conditions => cond}) 
+    opt.merge!({:include => {:artist => {}}, :order => 'pictures.updated_at desc'})
+    Picture.find(:all, opt)
+  end
+  
+  def self.list_as_json_text ary
+    '[' + ary.map {|i| i.to_json_with_picture_data }.join(',') + ']'
+  end
+  
+  def picture_data
+    Base64.encode64(self.restore)
+  end
+  
+  def to_json_with_picture_data
+    self.to_json({:methods => :picture_data})
   end
   
   def credit_template
-    "#{self.classname.tableize}/credit"
+    "#{self.classname.tableize}/attributes/credit"
+  end
+  
+  def full_credit_template
+    "#{self.classname.tableize}/attributes/full_credit"
   end
   
   def credit_data
@@ -122,10 +189,12 @@ class Picture < ActiveRecord::Base
   end
   
   def flags
+    return @flags if @flags
     begin
-      @flags = JSON.parse(self.settings) unless @flags
+      @flags = JSON.parse(self.settings)
     rescue 
     end
+    @flags = {} unless @flags
     @flags
   end
   
@@ -134,57 +203,57 @@ class Picture < ActiveRecord::Base
   end
   
   def flag_open
-    @flag_open = flags["open"] unless @flag_open
+    @flag_open = (flags["open"] || -1) unless @flag_open
     @flag_open
   end
   
   def flag_commercial
-    @flag_commercial = flags["commercial"] unless @flag_commercial
+    @flag_commercial = (flags["commercial"] || -1) unless @flag_commercial
     @flag_commercial
   end
   
   def flag_official
-    @flag_official = flags["official"] unless @flag_official
+    @flag_official = (flags["official"] || -1) unless @flag_official
     @flag_official
   end
   
   def flag_attribution
-    @flag_attribution = flags["attribution"] unless @flag_attribution
+    @flag_attribution = (flags["attribution"] || -1) unless @flag_attribution
     @flag_attribution
   end
   
   def flag_derive
-    @flag_derive = flags["derive"] unless @flag_derive
+    @flag_derive = (flags["derive"] || -1) unless @flag_derive
     @flag_derive
   end
   
   def flag_thumbnail
-    @flag_thumbnail = flags["thumbnail"] unless @flag_thumbnail
+    @flag_thumbnail = (flags["thumbnail"] || -1) unless @flag_thumbnail
     @flag_thumbnail
   end
   
   def flag_gif_convert
-    @flag_gif_convert = flags["gif_convert"] unless @flag_gif_convert
+    @flag_gif_convert = (flags["gif_convert"] || -1) unless @flag_gif_convert
     @flag_gif_convert
   end
   
   def flag_reverse
-    @flag_reverse = flags["reverse"] unless @flag_reverse
+    @flag_reverse = (flags["reverse"] || -1) unless @flag_reverse
     @flag_reverse
   end
   
   def flag_resize
-    @flag_resize = flags["resize"] unless @flag_resize
+    @flag_resize = (flags["resize"] || -1) unless @flag_resize
     @flag_resize
   end
   
   def flag_sync_vh
-    @flag_sync_vh = flags["sync_vh"] unless @flag_sync_vh
+    @flag_sync_vh = (flags["sync_vh"] || -1) unless @flag_sync_vh
     @flag_sync_vh
   end
   
   def flag_overlap
-    @flag_overlap = flags["overlap"] unless @flag_overlap
+    @flag_overlap = (flags["overlap"] || -1) unless @flag_overlap
     @flag_overlap
   end