require 'socket' require 'etc' module Fm DONT_PREVIEW_THESE_FILES = /\.(avi|[mj]pe?g|iso|mp\d|og[gmv]|wm[av]|mkv|torrent|so|class|flv|png|bmp|vob|divx?)$/i def column_put_file(n, file) i = 0 if Option.preview and Option.file_preview and file.path !~ DONT_PREVIEW_THESE_FILES m = lines - 2 attr_set(Color.base) left, wid = get_boundaries(n) if false and file.ext =~ /(?:rar|zip|7z|tar|gz)$/ and file.size < 10485760 text = `aunpack -l #{file.sh} 2>> /dev/null` text.each_line do |l| puti i+1, left, l[0, wid-1].ljust(wid) i += 1 break if i == m end else File.open(file.path, 'r') do |f| check = true left, wid = get_boundaries(n) f.lines.each do |l| if check check = false break unless l.each_byte.all? {|x| x.ord > 0 and x.ord < 128} end puti i+1, left, l.gsub("\t"," ")[0, wid-1].ljust(wid) i += 1 break if i == m end end end end column_clear(n, i) end def put_directory(c, d) l = 1 return column_clear(c, 0) unless d infos = (c == COLUMNS - 2) left, wid = get_boundaries(c) right = left + wid if not d.read? if (c == COLUMNS - 1) and @entering_directory # puti l, left, "reading...".ljust(wid+1) puti l, left, " " * (wid+1) column_clear(c, 1) @entering_directory = false end Scheduler << d return elsif d.read? and d.empty? puti l, left, 'empty'.ljust(wid+1) column_clear(c, 1) return end offset = get_offset(d, lines) (lines - 1).times do |l| lpo = l + offset l += 1 break if (f = d.files[lpo]) == nil mycolor = if lpo == d.pos if infos Color.selected_current_row else Color.selected end elsif f.marked? Color.marked else Color.normal end dir = false clrname = if f.symlink? dir = f.dir? if f.broken_symlink? :badlink else :goodlink end elsif f.dir? dir = true :directory elsif f.movie? :video elsif f.audio? :sound elsif f.image? :image elsif f.executable? :executable else :file end fn = f.displayname fn = "* #{fn}" if f.marked? if infos myinfo = " #{f.infostring} " sz = myinfo.size str = fn[0, wid-1].ljust(wid+1) if str.size > sz str[-sz..-1] = myinfo yes = true else yes = false end puti l, left, str attr_at(l, right-sz, sz, Color.normal.send(clrname)) else puti l, left, fn[0, wid-1].ljust(wid+1) end if infos and Option.wide_bar attr_at(l, left-1, wid+1, mycolor.send(clrname)) else attr_at(l, left, fn.size.limit(wid-1), mycolor.send(clrname)) end end column_clear(c, l-1) end def self.column_clear(n, from=0) attr_set(Color.base) left, wid = get_boundaries(n) (from -1).upto(lines) do |l| puti l+2, left, ' ' * (wid+1) end end def self.get_offset(dir, max) pos = dir.pos len = dir.files.size max -= 2 if len <= max or pos < max/2 return 0 elsif pos >= (len - max/2) return len - max else return pos - max/2 end end def self.get_boundaries(column) cols = CLI.cols # to cache case column when 0 return 0, cols / 8 when 1 q = cols / 8 if !Option.preview# or (!Option.filepreview and !currentfile.dir?) return q, 2*q # elsif currentfile.path != DONT_PREVIEW_THESE_FILES # return q, 2*q else return q, q end when 2 if !Option.preview q = cols * 0.375 - 1 w = @path.last.width.limit(cols * 0.625, cols/8) # elsif currentfile.path =~ DONT_PREVIEW_THESE_FILES or (!Option.filepreview and !currentfile.dir?) # q = cols / 4 # w = @path.last.width.limit(cols * 0.75, cols/8) else q = cols / 4 w = @path.last.width.limit(cols/2, cols/8) end return q, w when 3 l = cols / 4 + 1 l += @path.last.width.limit(cols/2, cols/8) return l, cols - l end end def self.draw attr_set(Color.base) @cur_y = get_boundaries(COLUMNS-2)[0] if @buffer =~ /^block/ screensaver elsif @buffer =~ /^\?/ cleari puti 0, " - - - Help - - -" if text = HELP[@buffer[1..-1]] i = 2 text.each_line do |l| break if i == lines puti(i, l) i += 1 end end elsif @buffer =~ /^o|`|'$/ cleari i = 0 case @buffer when 'o'; puti 0, " move files to:" when '`', "'"; puti 0, " go to:" end for key,val in @memory next if key == "'" break if key == lines puti i+=1, 2, "#{key} => #{val}" end else @pwd.make_sure_cursor_is_in_range cf = currentfile if cf and s0 = cf.mimetype puti 0, cols-s0.size, s0 end s1 = "#{Socket.gethostname}:" s2 = "#{@path.last.path.ascii_only_if(Option.ascii_only)}#{"/" unless @path.size == 1}" s3 = "#{cf ? cf.displayname : ''}" if s0 puti 0, (s1 + s2 + s3).ljust(cols-s0.size) else puti 0, (s1 + s2 + s3).ljust(cols) end bg = -1 attr_at(0, 0, s1.size, *Color.hostname) attr_at(0, s1.size, s2.size, *Color.currentdir) attr_at(0, s1.size + s2.size, s3.size, *Color.currentfile) # color_at 0, 0, -1, 7, bg # color_at 0, 0, s1.size, 7, bg # color_at 0, s1.size, s2.size, 6, bg # color_at 0, s1.size + s2.size, s3.size, 5, bg # bold false begin if Option.preview if cf.dir? dir = @dirs[ cf.path ] dir.sort_if_needed put_directory(3, dir) elsif cf.file? column_put_file(3, cf) else column_clear(3) end else column_clear(3) end rescue column_clear(3) end pos_constant = @path.size - COLUMNS + 1 (COLUMNS - 1).times do |c| pos = pos_constant + c if pos >= 0 put_directory(c, @path[pos]) else column_clear(c) end end attr_set(Color.base) btm = lines - 1 case @buffer when /^delete/, /^dd/ puti btm, "#@buffer ".rjust(cols) puti btm, 'Are you serious? (' + Option.confirm_string + ')' when 'S' puti btm, "Sort by (n)ame (s)ize (m)time (c)time (e)xtension mime(t)ype (CAPITAL:reversed)" when 't' puti btm, "Toggle (h)idden_files (d)irs_first (f)ilepreview (p)review (w)idebar (c)d (!)confirm" when 'e' puti btm, "Edit (a)pplications file(t)ypes" else owner = "#{Etc.getpwuid(cf.stat.uid).name}:#{Etc.getgrgid(cf.stat.gid).name}" attr_set(Color.base) attr_set(Color.info) puti btm, "#@buffer #{@pwd.file_size.bytes(false)}, #{@pwd.free_space.bytes(false)} free ".rjust(cols) more = '' if cf.symlink? more = "#{cf.readlink.ascii_only_if(Option.ascii_only)}" end attr_set(Color.date) left = " #{Time.now.strftime("%H:%M:%S %a %b %d")} " puti btm, left attr_set(cf.writable? ? Color.allowed : Color.denied) second = "#{cf.rights} #{owner} " puti btm, left.size, second if more attr_set(cf.exists? ? Color.allowed : Color.denied) puti btm, left.size + second.size, "#{more} " end end attr_set(Color.base) draw_bars unless @bars.empty? movi(@pwd.pos + 1 - get_offset(@pwd, lines), @cur_y) end CLI.refresh end def self.draw_bars() l = CLI.lines @bars.each do |bar| bar.update l -= 1 puti l, bar.text[0..cols-1].ljust(cols) done = bar.done c = (done * cols).to_i unless done == 0 attr_at(l, 0, c, *Color.bar_done) end unless done == cols attr_at(l, c, -1, *Color.bar_undone) end end end end