OSDN Git Service

t#31521:fix story pager
[pettanr/pettanr.git] / app / models / picture.rb
1 #実素材
2 class Picture < ActiveRecord::Base
3   belongs_to :original_picture
4   belongs_to :license
5   belongs_to :artist
6   has_one :resource_picture
7   
8   validates :original_picture_id, :presence => true, :numericality => true, :existence => {:both => false}
9   validates :revision, :presence => true, :numericality => true
10   validates :ext, :presence => true, :length => {:maximum => 4}, :inclusion => {:in => ['png', 'jpeg', 'gif']}
11   validates :width, :presence => true, :numericality => true, :natural_number => true
12   validates :height, :presence => true, :numericality => true, :natural_number => true
13   validates :filesize, :presence => true, :numericality => {:greater_than => 0, :less_than_or_equal_to => 2000000}, :natural_number => true
14   validates :md5, :presence => true, :length => {:minimum => 32, :maximum => 32}
15   validates :license_id, :presence => true, :numericality => true, :existence => {:both => false}
16   validates :artist_id, :presence => true, :numericality => true, :existence => {:both => false}
17   validates :artist_name, :presence => true
18   validates :classname, :presence => true, :length => {:maximum => 50}
19   
20   before_validation :valid_encode
21   
22   def valid_encode
23     ['artist_name', 'classname', 'credit', 'settings'].each do |a|
24       next if attributes[a] == nil
25       raise Pettanr::BadRequest unless attributes[a].valid_encoding?
26     end
27   end
28   
29   def supply_default
30   end
31   
32   def overwrite rp
33     attr = {:width => rp.width, :height => rp.height, :ext => rp.ext, :filesize => rp.filesize, 
34       :original_picture_id => rp.original_picture_id, :license_id => rp.license_id, :artist_id => rp.artist_id, 
35       :md5 => rp.md5, :artist_name => rp.artist_name, :classname => rp.classname, :credit => rp.credit, :settings => rp.settings
36     }
37     self.attributes = attr
38     self.revision = self.new_revision   #Do not move to attr. new_revision reffernces self.original_picture_id
39   end
40   
41   def own? roles
42     roles = [roles] unless roles.respond_to?(:each)
43     ar = Picture.get_artist_from_roles roles
44     return false unless ar
45     self.artist_id == ar.id
46   end
47   
48   def visible? ar
49     return true
50   end
51   
52   def showable? au = nil
53     return false unless self.original_picture
54     return true if self.own?(au)
55     self.enable? and self.head?
56   end
57   
58   def filename
59     "#{self.id}.#{self.ext}"
60   end
61   
62   def gifname
63     "#{self.id}.gif"
64   end
65   
66   def mime_type
67     "image/#{self.ext}"
68   end
69   
70   def url
71     '/pictures/' + filename
72   end
73   
74   def opt_img_tag
75     {:src => self.url, :width => self.width, :height => self.height}
76   end
77   
78   def tmb_opt_img_tag
79     tw, th = PettanImager.thumbnail_size(self.width, self.height)
80     {:src => self.url, :width => tw, :height => th}
81   end
82   
83   def tail_opt_img_tag img
84     {:src => img, :width => self.width, :height => self.height}
85   end
86   
87   def tail_tmb_opt_img_tag img
88     tw, th = PettanImager.thumbnail_size(self.width, self.height)
89     {:src => img, :width => tw, :height => th}
90   end
91   
92   def new_revision
93     Picture.maximum(:revision, :conditions => ['original_picture_id = ?', self.original_picture_id]).to_i + 1
94   end
95   
96   def enable?
97     r = self.head.resource_picture
98     r ? true : false
99   end
100   
101   def self.head opid
102     Picture.find(:first, :conditions => ['original_picture_id = ?', opid], :order => 'pictures.revision desc')
103   end
104   
105   def head
106     Picture.head(self.original_picture_id)
107   end
108   
109   def head?
110     self == head
111   end
112   
113   def to_gif?
114     self.ext == 'png' and self.flag_gif_convert >= 0
115   end
116   
117   def subdirs
118     flag_reverse < 0 ? [''] : ['', 'v', 'h', 'vh']
119   end
120   
121   def self.find_by_md5 md5
122     r = Picture.find :all, :conditions => ['pictures.md5 = ?', md5], :order => 'pictures.updated_at desc'
123   end
124   
125   def self.list_by_md5 md5, opid = nil
126     cond = if opid.blank?
127       ['pictures.md5 = ?', md5]
128     else
129       ['pictures.md5 = ? and pictures.original_picture_id <> ?', md5, opid]
130     end
131     r = Picture.find :all, :conditions => cond, :order => 'pictures.updated_at desc'
132   end
133   
134   def self.exist_by_md5 md5, opid
135     Picture.list_by_md5(md5, opid).empty? ? false : true
136   end
137   
138   def self.show rid, roles
139     opt = {}
140     r = Picture.find(rid, opt)
141     raise ActiveRecord::Forbidden unless r.visible?(roles)
142     r
143   end
144   
145   def store(imager)
146     res = false
147     if res = self.save
148       if res = self.store_picture(imager, self.filename)
149         if self.to_gif?
150           if gifimager = imager.to_gif
151             res = self.store_picture(gifimager, self.gifname)
152           else
153             res = false
154           end
155         end
156       end
157     end
158     res
159   end
160   
161   def store_picture(imager, fn)
162     res = false
163     subdirs.each do |d|
164       picdata = d.empty? ? imager.binary : imager.__send__(d)
165       res = PictureIO.picture_io.put(picdata, fn, d)
166       break unless res
167     end
168     res
169   end
170   
171   def restore(subdir = nil)
172     PictureIO.picture_io.get self.filename, subdir
173   end
174   
175   def self.export(dt = nil)
176     opt = {}
177     cond = if dt
178       ['artists.author_id is not null and pictures.updated_at >= ?', dt]
179     else
180       'artists.author_id is not null'
181     end
182     opt.merge!({:conditions => cond}) 
183     opt.merge!({:include => {:artist => {}}, :order => 'pictures.updated_at desc'})
184     Picture.find(:all, opt)
185   end
186   
187   def self.list_as_json_text ary
188     '[' + ary.map {|i| i.to_json_with_picture_data }.join(',') + ']'
189   end
190   
191   def picture_data
192     Base64.encode64(self.restore)
193   end
194   
195   def to_json_with_picture_data
196     self.to_json({:methods => :picture_data})
197   end
198   
199   def unpublish
200     imager = PettanImager.load(File.open(Rails.root + 'app/assets/images/error.png', 'rb').read)
201     return false unless imager
202     self.store imager
203   end
204   
205   def credit_template
206     "#{self.classname.tableize}/attributes/credit"
207   end
208   
209   def full_credit_template
210     "#{self.classname.tableize}/attributes/full_credit"
211   end
212   
213   def credit_data
214     begin
215       @credit_data = JSON.parse(self.credit) unless @credit_data
216     rescue 
217     end
218     @credit_data
219   end
220   
221   def flags
222     return @flags if @flags
223     begin
224       @flags = JSON.parse(self.settings)
225     rescue 
226     end
227     @flags = {} unless @flags
228     @flags
229   end
230   
231   def flags=(s)
232     @flags = s
233   end
234   
235   def flag_open
236     @flag_open = (flags["open"] || -1) unless @flag_open
237     @flag_open
238   end
239   
240   def flag_commercial
241     @flag_commercial = (flags["commercial"] || -1) unless @flag_commercial
242     @flag_commercial
243   end
244   
245   def flag_official
246     @flag_official = (flags["official"] || -1) unless @flag_official
247     @flag_official
248   end
249   
250   def flag_attribution
251     @flag_attribution = (flags["attribution"] || -1) unless @flag_attribution
252     @flag_attribution
253   end
254   
255   def flag_derive
256     @flag_derive = (flags["derive"] || -1) unless @flag_derive
257     @flag_derive
258   end
259   
260   def flag_thumbnail
261     @flag_thumbnail = (flags["thumbnail"] || -1) unless @flag_thumbnail
262     @flag_thumbnail
263   end
264   
265   def flag_gif_convert
266     @flag_gif_convert = (flags["gif_convert"] || -1) unless @flag_gif_convert
267     @flag_gif_convert
268   end
269   
270   def flag_reverse
271     @flag_reverse = (flags["reverse"] || -1) unless @flag_reverse
272     @flag_reverse
273   end
274   
275   def flag_resize
276     @flag_resize = (flags["resize"] || -1) unless @flag_resize
277     @flag_resize
278   end
279   
280   def flag_sync_vh
281     @flag_sync_vh = (flags["sync_vh"] || -1) unless @flag_sync_vh
282     @flag_sync_vh
283   end
284   
285   def flag_overlap
286     @flag_overlap = (flags["overlap"] || -1) unless @flag_overlap
287     @flag_overlap
288   end
289   
290 end