OSDN Git Service

t#32402:add file prof
[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 symbol_option
93     self.tmb_opt_img_tag
94   end
95   
96   def new_revision
97     Picture.maximum(:revision, :conditions => ['original_picture_id = ?', self.original_picture_id]).to_i + 1
98   end
99   
100   def enable?
101     r = self.head.resource_picture
102     r ? true : false
103   end
104   
105   def self.head opid
106     Picture.find(:first, :conditions => ['original_picture_id = ?', opid], :order => 'pictures.revision desc')
107   end
108   
109   def head
110     Picture.head(self.original_picture_id)
111   end
112   
113   def head?
114     self == head
115   end
116   
117   def to_gif?
118     self.ext == 'png' and self.flag_gif_convert >= 0
119   end
120   
121   def subdirs
122     flag_reverse < 0 ? [''] : ['', 'v', 'h', 'vh']
123   end
124   
125   def self.find_by_md5 md5
126     r = Picture.find :all, :conditions => ['pictures.md5 = ?', md5], :order => 'pictures.updated_at desc'
127   end
128   
129   def self.list_by_md5 md5, opid = nil
130     cond = if opid.blank?
131       ['pictures.md5 = ?', md5]
132     else
133       ['pictures.md5 = ? and pictures.original_picture_id <> ?', md5, opid]
134     end
135     r = Picture.find :all, :conditions => cond, :order => 'pictures.updated_at desc'
136   end
137   
138   def self.exist_by_md5 md5, opid
139     Picture.list_by_md5(md5, opid).empty? ? false : true
140   end
141   
142   def self.show rid, roles
143     opt = {}
144     r = Picture.find(rid, opt)
145     raise ActiveRecord::Forbidden unless r.visible?(roles)
146     r
147   end
148   
149   def store(imager)
150     res = false
151     if res = self.save
152       if res = self.store_picture(imager, self.filename)
153         if self.to_gif?
154           if gifimager = imager.to_gif
155             res = self.store_picture(gifimager, self.gifname)
156           else
157             res = false
158           end
159         end
160       end
161     end
162     res
163   end
164   
165   def store_picture(imager, fn)
166     res = false
167     subdirs.each do |d|
168       picdata = d.empty? ? imager.binary : imager.__send__(d)
169       res = PictureIO.picture_io.put(picdata, fn, d)
170       break unless res
171     end
172     res
173   end
174   
175   def restore(subdir = nil)
176     PictureIO.picture_io.get self.filename, subdir
177   end
178   
179   def self.export(dt = nil)
180     opt = {}
181     cond = if dt
182       ['artists.author_id is not null and pictures.updated_at >= ?', dt]
183     else
184       'artists.author_id is not null'
185     end
186     opt.merge!({:conditions => cond}) 
187     opt.merge!({:include => {:artist => {}}, :order => 'pictures.updated_at desc'})
188     Picture.find(:all, opt)
189   end
190   
191   def self.list_as_json_text ary
192     '[' + ary.map {|i| i.to_json_with_picture_data }.join(',') + ']'
193   end
194   
195   def picture_data
196     Base64.encode64(self.restore)
197   end
198   
199   def to_json_with_picture_data
200     self.to_json({:methods => :picture_data})
201   end
202   
203   def unpublish
204     imager = PettanImager.load(File.open(Rails.root + 'app/assets/images/error.png', 'rb').read)
205     return false unless imager
206     self.store imager
207   end
208   
209   def credit_template
210     "#{self.classname.tableize}/attributes/credit"
211   end
212   
213   def full_credit_template
214     "#{self.classname.tableize}/attributes/full_credit"
215   end
216   
217   def credit_data
218     begin
219       @credit_data = JSON.parse(self.credit) unless @credit_data
220     rescue 
221     end
222     @credit_data
223   end
224   
225   def flags
226     return @flags if @flags
227     begin
228       @flags = JSON.parse(self.settings)
229     rescue 
230     end
231     @flags = {} unless @flags
232     @flags
233   end
234   
235   def flags=(s)
236     @flags = s
237   end
238   
239   def flag_open
240     @flag_open = (flags["open"] || -1) unless @flag_open
241     @flag_open
242   end
243   
244   def flag_commercial
245     @flag_commercial = (flags["commercial"] || -1) unless @flag_commercial
246     @flag_commercial
247   end
248   
249   def flag_official
250     @flag_official = (flags["official"] || -1) unless @flag_official
251     @flag_official
252   end
253   
254   def flag_attribution
255     @flag_attribution = (flags["attribution"] || -1) unless @flag_attribution
256     @flag_attribution
257   end
258   
259   def flag_derive
260     @flag_derive = (flags["derive"] || -1) unless @flag_derive
261     @flag_derive
262   end
263   
264   def flag_thumbnail
265     @flag_thumbnail = (flags["thumbnail"] || -1) unless @flag_thumbnail
266     @flag_thumbnail
267   end
268   
269   def flag_gif_convert
270     @flag_gif_convert = (flags["gif_convert"] || -1) unless @flag_gif_convert
271     @flag_gif_convert
272   end
273   
274   def flag_reverse
275     @flag_reverse = (flags["reverse"] || -1) unless @flag_reverse
276     @flag_reverse
277   end
278   
279   def flag_resize
280     @flag_resize = (flags["resize"] || -1) unless @flag_resize
281     @flag_resize
282   end
283   
284   def flag_sync_vh
285     @flag_sync_vh = (flags["sync_vh"] || -1) unless @flag_sync_vh
286     @flag_sync_vh
287   end
288   
289   def flag_overlap
290     @flag_overlap = (flags["overlap"] || -1) unless @flag_overlap
291     @flag_overlap
292   end
293   
294 end