2012-01-02 16:44:10 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# vi:si:et:sw=4:sts=4:ts=4
|
2014-09-30 19:04:46 +00:00
|
|
|
from __future__ import with_statement, division, print_function
|
2012-01-02 16:44:10 +00:00
|
|
|
import chardet
|
|
|
|
import re
|
2012-03-02 21:37:35 +00:00
|
|
|
import codecs
|
2012-01-02 16:44:10 +00:00
|
|
|
|
|
|
|
import ox
|
|
|
|
|
|
|
|
|
|
|
|
__all__ = []
|
|
|
|
|
|
|
|
|
2012-08-14 14:12:43 +00:00
|
|
|
def _detect_encoding(fp):
|
2012-01-02 16:44:10 +00:00
|
|
|
bomDict={ # bytepattern : name
|
|
|
|
(0x00, 0x00, 0xFE, 0xFF): "utf_32_be",
|
|
|
|
(0xFF, 0xFE, 0x00, 0x00): "utf_32_le",
|
|
|
|
(0xFE, 0xFF, None, None): "utf_16_be",
|
|
|
|
(0xFF, 0xFE, None, None): "utf_16_le",
|
|
|
|
(0xEF, 0xBB, 0xBF, None): "utf_8",
|
|
|
|
}
|
|
|
|
|
|
|
|
# go to beginning of file and get the first 4 bytes
|
|
|
|
oldFP = fp.tell()
|
|
|
|
fp.seek(0)
|
|
|
|
(byte1, byte2, byte3, byte4) = tuple(map(ord, fp.read(4)))
|
|
|
|
|
|
|
|
# try bom detection using 4 bytes, 3 bytes, or 2 bytes
|
|
|
|
bomDetection = bomDict.get((byte1, byte2, byte3, byte4))
|
|
|
|
if not bomDetection:
|
|
|
|
bomDetection = bomDict.get((byte1, byte2, byte3, None))
|
|
|
|
if not bomDetection:
|
|
|
|
bomDetection = bomDict.get((byte1, byte2, None, None))
|
|
|
|
## if BOM detected, we're done :-)
|
|
|
|
fp.seek(oldFP)
|
|
|
|
if bomDetection:
|
|
|
|
return bomDetection
|
|
|
|
encoding = 'latin-1'
|
|
|
|
#more character detecting magick using http://chardet.feedparser.org/
|
|
|
|
fp.seek(0)
|
|
|
|
rawdata = fp.read()
|
2012-01-28 08:26:42 +00:00
|
|
|
#if data can be decoded as utf-8 use that, try chardet otherwise
|
|
|
|
#chardet detects utf-8 as ISO-8859-2 most of the time
|
|
|
|
try:
|
|
|
|
data = unicode(rawdata, 'utf-8')
|
|
|
|
encoding = 'utf-8'
|
|
|
|
except:
|
|
|
|
encoding = chardet.detect(rawdata)['encoding']
|
2012-01-02 16:44:10 +00:00
|
|
|
fp.seek(oldFP)
|
|
|
|
return encoding
|
|
|
|
|
|
|
|
|
|
|
|
def load(filename, offset=0):
|
|
|
|
'''
|
|
|
|
filename path to an srt file
|
|
|
|
offset in seconds shift all in/out points by offset
|
|
|
|
|
|
|
|
returns list with objects that have in,out,value and id
|
|
|
|
'''
|
|
|
|
srt = []
|
|
|
|
|
|
|
|
def parse_time(t):
|
|
|
|
return offset + ox.time2ms(t.replace(',', '.')) / 1000
|
|
|
|
|
|
|
|
with open(filename) as f:
|
2012-08-14 14:12:43 +00:00
|
|
|
encoding = _detect_encoding(f)
|
2012-01-02 16:44:10 +00:00
|
|
|
data = f.read()
|
|
|
|
try:
|
|
|
|
data = unicode(data, encoding)
|
|
|
|
except:
|
|
|
|
try:
|
|
|
|
data = unicode(data, 'latin-1')
|
|
|
|
except:
|
2014-09-30 19:04:46 +00:00
|
|
|
print("failed to detect encoding, giving up")
|
2012-01-02 16:44:10 +00:00
|
|
|
return srt
|
|
|
|
|
|
|
|
data = data.replace('\r\n', '\n')
|
2012-09-21 09:48:02 +00:00
|
|
|
srts = re.compile('(\d\d:\d\d:\d\d[,.]\d\d\d)\s*?-->\s*?(\d\d:\d\d:\d\d[,.]\d\d\d).*?\n(.*?)\n\n', re.DOTALL)
|
2012-01-02 16:44:10 +00:00
|
|
|
i = 0
|
|
|
|
for s in srts.findall(data):
|
|
|
|
_s = {'id': str(i),
|
|
|
|
'in': parse_time(s[0]),
|
|
|
|
'out': parse_time(s[1]),
|
|
|
|
'value': s[2].strip()
|
|
|
|
}
|
|
|
|
srt.append(_s)
|
|
|
|
i += 1
|
|
|
|
return srt
|
2012-01-02 17:07:17 +00:00
|
|
|
|
|
|
|
def encode(data):
|
|
|
|
'''
|
|
|
|
encodes list of objects with in,out,value into srt
|
|
|
|
result is utf-8 encoded bytestring
|
|
|
|
'''
|
|
|
|
srt = u''
|
|
|
|
i = 1
|
|
|
|
for s in data:
|
|
|
|
srt += '%d\r\n%s --> %s\r\n%s\r\n\r\n' % (
|
|
|
|
i,
|
2012-08-14 13:58:05 +00:00
|
|
|
ox.format_duration(s['in']*1000, years=False).replace('.', ','),
|
|
|
|
ox.format_duration(s['out']*1000, years=False).replace('.', ','),
|
2012-01-02 17:07:17 +00:00
|
|
|
s['value'].replace('\n', '\r\n').strip()
|
|
|
|
)
|
|
|
|
i += 1
|
2012-03-02 21:37:35 +00:00
|
|
|
return codecs.BOM_UTF8 + srt.encode('utf-8')
|
2012-01-02 17:07:17 +00:00
|
|
|
|