2008-11-12 23:33:32 -05:00
|
|
|
import commands, dirutil, grp, mailbox, method, mode, os.path, pwd, re, sys
|
2008-12-12 22:46:06 -05:00
|
|
|
import buffer, default, util, window
|
2008-11-12 23:33:32 -05:00
|
|
|
from mode.mutt import MuttGrammar
|
|
|
|
from lex import Grammar, PatternRule, RegionRule, PatternGroupRule
|
|
|
|
from point import Point
|
|
|
|
from buffer import Buffer
|
|
|
|
from method import Method, Argument, arg
|
|
|
|
|
|
|
|
try:
|
|
|
|
import eyeD3
|
|
|
|
has_eyed3 = True
|
|
|
|
except ImportError:
|
|
|
|
has_eyed3 = False
|
|
|
|
|
2008-11-13 18:19:19 -05:00
|
|
|
def _size_info(i):
|
|
|
|
if i >= 1024 ** 2:
|
|
|
|
return '%.2f MB' % (i * 1.0 / 1024 ** 2)
|
|
|
|
elif i >= 1024:
|
|
|
|
return '%.2f KB' % (i * 1.0 / 1024)
|
|
|
|
else:
|
|
|
|
return '%d B' % i
|
|
|
|
|
|
|
|
def _time_info(i):
|
|
|
|
if i >= 60 ** 2:
|
|
|
|
(j, k) = (i // 60 ** 2, i % 60 ** 2)
|
|
|
|
return '%d:%02d:%02d secs' % (j, k // 60, k % 60)
|
|
|
|
elif i >= 60:
|
|
|
|
return '%d:%02d secs' % (i // 60, i % 60)
|
|
|
|
else:
|
|
|
|
return '%d secs' % i
|
|
|
|
|
2008-11-12 23:33:32 -05:00
|
|
|
def _str_info(s):
|
|
|
|
if s:
|
|
|
|
return str(s)
|
|
|
|
else:
|
|
|
|
return ''
|
2008-11-13 18:19:19 -05:00
|
|
|
|
2008-11-12 23:33:32 -05:00
|
|
|
def _num_pair_info(t):
|
|
|
|
if t[0] and t[1]:
|
|
|
|
return '%s/%s' % t
|
|
|
|
elif t[0]:
|
|
|
|
return str(t[0])
|
|
|
|
elif t[1]:
|
|
|
|
return '/%s' % t[1]
|
|
|
|
else:
|
|
|
|
return ''
|
2008-11-13 18:19:19 -05:00
|
|
|
|
2008-11-12 23:33:32 -05:00
|
|
|
class Mp3Buffer(Buffer):
|
|
|
|
btype = 'mp3'
|
|
|
|
def __init__(self, path):
|
|
|
|
Buffer.__init__(self)
|
2008-12-12 22:46:06 -05:00
|
|
|
self.path = util.expand_tilde(path)
|
2008-11-12 23:33:32 -05:00
|
|
|
self.base = os.path.basename(self.path)
|
2008-11-13 18:19:19 -05:00
|
|
|
self.af = eyeD3.tag.Mp3AudioFile(self.path)
|
|
|
|
self.tag = self.af.getTag()
|
2008-11-12 23:33:32 -05:00
|
|
|
def changed(self): return False
|
|
|
|
def readonly(self): return True
|
|
|
|
def path_exists(self): raise Exception
|
|
|
|
def _make_path(self, name): raise Exception
|
|
|
|
def save(self, force=False): raise Exception, "can't save an mbox"
|
|
|
|
def save_as(self, path): raise Exception, "can't save an mbox"
|
|
|
|
def name(self): return 'mp3:%s' % (self.base)
|
2008-11-13 18:19:19 -05:00
|
|
|
def open(self): self.lines = self._get_lines()
|
|
|
|
def reload(self): self.set_lines(self._get_lines(), force=True)
|
2008-11-12 23:33:32 -05:00
|
|
|
def _get_lines(self):
|
|
|
|
lines = [
|
|
|
|
'# INFO',
|
2008-11-13 18:19:19 -05:00
|
|
|
'Filename: ' + os.path.basename(self.af.fileName),
|
|
|
|
'Filesize: ' + _size_info(self.af.fileSize),
|
|
|
|
'Playtime: ' + _time_info(self.af.getPlayTime()),
|
|
|
|
'Bitrate: ' + _str_info(self.af.getBitRate()),
|
|
|
|
'Samplerate: ' + _str_info(self.af.getSampleFreq()) + ' kHz',
|
2008-11-12 23:33:32 -05:00
|
|
|
'# ID3',
|
|
|
|
'Artist: ' + _str_info(self.tag.getArtist()),
|
|
|
|
'Album: ' + _str_info(self.tag.getAlbum()),
|
|
|
|
'BPM: ' + _str_info(self.tag.getBPM()),
|
|
|
|
#'CDID: ' + str(self.tag.getCDID()), #xyz
|
|
|
|
'Comment: ' + _str_info(self.tag.getComment()),
|
|
|
|
'DiscNum: ' + _num_pair_info(self.tag.getDiscNum()),
|
|
|
|
'Genre: ' + _str_info(self.tag.getGenre()),
|
|
|
|
'Publisher: ' + _str_info(self.tag.getPublisher()),
|
|
|
|
'Title: ' + _str_info(self.tag.getTitle()),
|
|
|
|
'TrackNum: ' + _num_pair_info(self.tag.getTrackNum()),
|
|
|
|
#'Version: ' + _str_info(self.tag.getVersion()),
|
|
|
|
'Year: ' + _str_info(self.tag.getYear()),
|
|
|
|
]
|
|
|
|
return lines
|
|
|
|
|
|
|
|
class TagGrammar(Grammar):
|
|
|
|
rules = [
|
|
|
|
PatternRule(r'data', r'[^\\\n]+'),
|
|
|
|
PatternRule(r'escaped', r'\\x[0-9a-fA-F]{2}'),
|
|
|
|
]
|
|
|
|
class Mp3Grammar(Grammar):
|
|
|
|
rules = [
|
|
|
|
PatternRule(r'comment', r'#.*\n$'),
|
|
|
|
PatternRule(r'mp3_label', r'^[^:]+(?=: )'),
|
|
|
|
RegionRule(r'mp3_tag', r': ', TagGrammar, r'\n$'),
|
|
|
|
]
|
|
|
|
|
|
|
|
class Mp3Refresh(Method):
|
|
|
|
def _execute(self, w, **vargs):
|
|
|
|
w.buffer.reload()
|
|
|
|
|
|
|
|
class Mp3OpenPath(Method):
|
|
|
|
args = [arg('mp3', dt="path", p="Open MP3: ", dv=default.path_dirname,
|
|
|
|
ld=True, h="mp3 to open")]
|
|
|
|
def _execute(self, w, **vargs):
|
|
|
|
b = Mp3Buffer(vargs['mp3'])
|
|
|
|
b.open()
|
|
|
|
window.Window(b, w.application, height=0, width=0, mode_name='mp3')
|
|
|
|
w.application.add_buffer(b)
|
|
|
|
w.application.switch_buffer(b)
|
|
|
|
|
|
|
|
class Mp3(mode.Fundamental):
|
|
|
|
modename = 'Mp3'
|
|
|
|
grammar = Mp3Grammar
|
|
|
|
actions = [Mp3Refresh, Mp3OpenPath]
|
|
|
|
colors = {
|
|
|
|
'mp3_label': ('blue', 'default', 'bold'),
|
|
|
|
'mp3_tag.start': ('default', 'default', 'bold'),
|
|
|
|
'mp3_tag.data': ('green', 'default', 'bold'),
|
|
|
|
'mp3_tag.escaped': ('magenta', 'default', 'bold'),
|
|
|
|
}
|
|
|
|
def __init__(self, w):
|
|
|
|
mode.Fundamental.__init__(self, w)
|
|
|
|
self.add_bindings('mp3-refresh', ('C-c r',))
|
|
|
|
|
|
|
|
if has_eyed3:
|
|
|
|
install = Mp3.install
|
|
|
|
else:
|
|
|
|
install = lambda *args: None
|