440 lines
18 KiB
JavaScript
440 lines
18 KiB
JavaScript
"use strict";
|
||
|
||
function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
|
||
function _slicedToArray(r, e) { return _arrayWithHoles(r) || _iterableToArrayLimit(r, e) || _unsupportedIterableToArray(r, e) || _nonIterableRest(); }
|
||
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
|
||
function _iterableToArrayLimit(r, l) { var t = null == r ? null : "undefined" != typeof Symbol && r[Symbol.iterator] || r["@@iterator"]; if (null != t) { var e, n, i, u, a = [], f = !0, o = !1; try { if (i = (t = t.call(r)).next, 0 === l) { if (Object(t) !== t) return; f = !1; } else for (; !(f = (e = i.call(t)).done) && (a.push(e.value), a.length !== l); f = !0); } catch (r) { o = !0, n = r; } finally { try { if (!f && null != t["return"] && (u = t["return"](), Object(u) !== u)) return; } finally { if (o) throw n; } } return a; } }
|
||
function _arrayWithHoles(r) { if (Array.isArray(r)) return r; }
|
||
function _createForOfIteratorHelper(r, e) { var t = "undefined" != typeof Symbol && r[Symbol.iterator] || r["@@iterator"]; if (!t) { if (Array.isArray(r) || (t = _unsupportedIterableToArray(r)) || e && r && "number" == typeof r.length) { t && (r = t); var _n = 0, F = function F() {}; return { s: F, n: function n() { return _n >= r.length ? { done: !0 } : { done: !1, value: r[_n++] }; }, e: function e(r) { throw r; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var o, a = !0, u = !1; return { s: function s() { t = t.call(r); }, n: function n() { var r = t.next(); return a = r.done, r; }, e: function e(r) { u = !0, o = r; }, f: function f() { try { a || null == t["return"] || t["return"](); } finally { if (u) throw o; } } }; }
|
||
function _unsupportedIterableToArray(r, a) { if (r) { if ("string" == typeof r) return _arrayLikeToArray(r, a); var t = {}.toString.call(r).slice(8, -1); return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0; } }
|
||
function _arrayLikeToArray(r, a) { (null == a || a > r.length) && (a = r.length); for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e]; return n; }
|
||
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
||
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
|
||
function _classCallCheck(a, n) { if (!(a instanceof n)) throw new TypeError("Cannot call a class as a function"); }
|
||
function _defineProperties(e, r) { for (var t = 0; t < r.length; t++) { var o = r[t]; o.enumerable = o.enumerable || !1, o.configurable = !0, "value" in o && (o.writable = !0), Object.defineProperty(e, _toPropertyKey(o.key), o); } }
|
||
function _createClass(e, r, t) { return r && _defineProperties(e.prototype, r), t && _defineProperties(e, t), Object.defineProperty(e, "prototype", { writable: !1 }), e; }
|
||
function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
|
||
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : i + ""; }
|
||
function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
|
||
/*字幕处理*/
|
||
var YbSubtitle = /*#__PURE__*/function () {
|
||
function YbSubtitle(player, list, config) {
|
||
_classCallCheck(this, YbSubtitle);
|
||
this.player = player;
|
||
this.list = list || [];
|
||
this.config = config || {};
|
||
this.activeIndex = -1; //渲染字幕位置索引
|
||
this.paused = true; //停止渲染
|
||
this._animation = null; //渲染动画
|
||
}
|
||
//默认配置
|
||
return _createClass(YbSubtitle, [{
|
||
key: "unload",
|
||
value: function unload() {
|
||
this.pause();
|
||
var container = this.player.container;
|
||
var div = container.getElementsByClassName('yb-player-subtitle-text')[0];
|
||
if (div) div.remove();
|
||
}
|
||
}, {
|
||
key: "setConfig",
|
||
value: function setConfig(key, value) {
|
||
this.config[key] = value;
|
||
}
|
||
}, {
|
||
key: "play",
|
||
value: function play() {
|
||
this.paused = false;
|
||
this._render();
|
||
}
|
||
}, {
|
||
key: "pause",
|
||
value: function pause() {
|
||
this.paused = true;
|
||
if (this._animation) {
|
||
window.cancelAnimationFrame(this._animation);
|
||
this._animation = null;
|
||
}
|
||
}
|
||
//渲染字幕
|
||
}, {
|
||
key: "_render",
|
||
value: function _render() {
|
||
var _this = this;
|
||
var container = this.player.container;
|
||
var video = this.player.video;
|
||
if (!video || !container) return;
|
||
var div = container.getElementsByClassName('yb-player-subtitle-text')[0];
|
||
var wrapperEl = container.getElementsByClassName('yb-player-wrapper')[0];
|
||
var nowIndex = this.list.findIndex(function (item) {
|
||
return YbPlayer.timeToSeconds(item.startTime) <= video.currentTime && YbPlayer.timeToSeconds(item.endTime) >= video.currentTime;
|
||
});
|
||
var config = _objectSpread(_objectSpread({}, YbSubtitle.DEFAULT_CONFIG), this.config);
|
||
if (nowIndex > -1 && this.activeIndex != nowIndex) {
|
||
this.activeIndex = nowIndex;
|
||
var nowTitle = this.list[nowIndex];
|
||
if (!div) {
|
||
div = document.createElement('DIV');
|
||
div.setAttribute('class', 'yb-player-subtitle-text');
|
||
wrapperEl.appendChild(div);
|
||
}
|
||
div.style.color = config.color;
|
||
div.style.fontSize = config.fontSize + 'px';
|
||
div.style.bottom = config.bottom;
|
||
div.style.textShadow = "0 0 5px " + config.shadowColor;
|
||
div.innerHTML = "<span>".concat(nowTitle.text, "</span>");
|
||
} else {
|
||
if (div && nowIndex == -1) div.remove();
|
||
}
|
||
if (!this.paused) this._animation = window.requestAnimationFrame(function () {
|
||
return _this._render();
|
||
});
|
||
}
|
||
}], [{
|
||
key: "init",
|
||
value: function init(player, src, config) {
|
||
return new Promise(function (resolve) {
|
||
YbSubtitle.showLoading(player);
|
||
var xhr = new XMLHttpRequest();
|
||
xhr.onreadystatechange = function () {
|
||
if (xhr.readyState == 4) {
|
||
if (xhr.status == 200) {
|
||
YbSubtitle.showResult(player, '加载字幕文件成功');
|
||
var list = src.includes('.srt') ? YbSubtitle.parseSrt(xhr.responseText) : src.includes('.ass') ? YbSubtitle.parseAss(xhr.responseText).events.data.map(function (item) {
|
||
var text = item.Text.replace(/\{.*?\}/g, ''); // 移除{}标签
|
||
text = text.replace(/\\N/g, '<br>'); // 处理换行
|
||
return _objectSpread(_objectSpread({}, item), {}, {
|
||
text: text,
|
||
startTime: item.Start,
|
||
endTime: item.End
|
||
});
|
||
}) : YbSubtitle.parseVtt(xhr.responseText).cues;
|
||
resolve(new YbSubtitle(player, list, config));
|
||
} else {
|
||
YbSubtitle.showResult(player, '加载字幕文件失败');
|
||
resolve(null);
|
||
}
|
||
xhr.abort();
|
||
}
|
||
};
|
||
xhr.onabort = function () {
|
||
xhr = null;
|
||
};
|
||
xhr.open('GET', src);
|
||
xhr.responseType = 'text';
|
||
xhr.send();
|
||
});
|
||
}
|
||
}, {
|
||
key: "showLoading",
|
||
value: function showLoading(player) {
|
||
var container = player.container;
|
||
var wrapperEl = container ? container.getElementsByClassName('yb-player-wrapper')[0] : null;
|
||
if (wrapperEl) {
|
||
var div = document.createElement('DIV');
|
||
div.setAttribute('class', 'yb-player-subtitle-loading');
|
||
div.innerHTML = '正在加载字幕文件';
|
||
wrapperEl.appendChild(div);
|
||
}
|
||
}
|
||
}, {
|
||
key: "hideLoading",
|
||
value: function hideLoading(player) {
|
||
var container = player === null || player === void 0 ? void 0 : player.container;
|
||
var loadingEl = container ? container.getElementsByClassName('yb-player-subtitle-loading')[0] : null;
|
||
if (loadingEl) loadingEl.remove();
|
||
}
|
||
}, {
|
||
key: "showResult",
|
||
value: function showResult(player, message) {
|
||
var container = player.container;
|
||
var loadingEl = container ? container.getElementsByClassName('yb-player-subtitle-loading')[0] : null;
|
||
if (loadingEl) loadingEl.innerHTML = message;
|
||
window.setTimeout(function () {
|
||
YbSubtitle.hideLoading(player);
|
||
}, 1000);
|
||
}
|
||
//格式化SRT字幕
|
||
}, {
|
||
key: "parseSrt",
|
||
value: function parseSrt(content) {
|
||
// 按空行分割字幕块
|
||
var blocks = content.trim().split(/\n\s*\n/);
|
||
var result = [];
|
||
for (var i = 0; i < blocks.length; i++) {
|
||
var lines = blocks[i].split('\n').filter(function (line) {
|
||
return line.trim() !== '';
|
||
});
|
||
if (lines.length < 3) {
|
||
continue; // 跳过无效块
|
||
}
|
||
|
||
// 解析序号
|
||
var index = parseInt(lines[0]);
|
||
if (isNaN(index)) {
|
||
throw new Error("\u65E0\u6548\u7684\u5E8F\u53F7: ".concat(lines[0]));
|
||
}
|
||
|
||
// 解析时间码
|
||
var timecodeMatch = lines[1].match(/(\d{2}):(\d{2}):(\d{2}),(\d{3})\s*-->\s*(\d{2}):(\d{2}):(\d{2}),(\d{3})/);
|
||
if (!timecodeMatch) {
|
||
throw new Error("\u65E0\u6548\u7684\u65F6\u95F4\u7801\u683C\u5F0F: ".concat(lines[1]));
|
||
}
|
||
var startTime = "".concat(parseInt(timecodeMatch[1]), ":").concat(parseInt(timecodeMatch[2]), ":").concat(parseInt(timecodeMatch[3]), ".").concat(parseInt(timecodeMatch[4]));
|
||
var endTime = "".concat(parseInt(timecodeMatch[5]), ":").concat(parseInt(timecodeMatch[6]), ":").concat(parseInt(timecodeMatch[7]), ".").concat(parseInt(timecodeMatch[8]));
|
||
|
||
// 合并文本行
|
||
var text = lines.slice(2).join('\n');
|
||
result.push({
|
||
index: index,
|
||
startTime: startTime,
|
||
endTime: endTime,
|
||
text: text
|
||
});
|
||
}
|
||
return result;
|
||
}
|
||
}, {
|
||
key: "parseAss",
|
||
value: function parseAss(content) {
|
||
var lines = content.split('\n');
|
||
var currentSection = '';
|
||
var result = {
|
||
info: {},
|
||
styles: {
|
||
format: [],
|
||
data: []
|
||
},
|
||
events: {
|
||
format: [],
|
||
data: []
|
||
}
|
||
};
|
||
for (var i = 0; i < lines.length; i++) {
|
||
var line = lines[i].trim();
|
||
|
||
// 跳过空行和注释
|
||
if (!line || line.startsWith(';')) continue;
|
||
|
||
// 检测段落开始
|
||
if (line.startsWith('[') && line.endsWith(']')) {
|
||
currentSection = line.slice(1, -1).toLowerCase();
|
||
continue;
|
||
}
|
||
// 根据当前段落处理内容
|
||
switch (currentSection) {
|
||
case 'script info':
|
||
parseScriptInfo(line, result.info);
|
||
break;
|
||
case 'v4+ styles':
|
||
parseStyles(line, result.styles);
|
||
break;
|
||
case 'events':
|
||
parseEvents(line, result.events);
|
||
break;
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
}, {
|
||
key: "parseVtt",
|
||
value: function parseVtt(content) {
|
||
var lines = content.split('\n');
|
||
var cues = [];
|
||
var currentCue = null;
|
||
var note = null;
|
||
var style = null;
|
||
var region = null;
|
||
|
||
// 检查文件头
|
||
if (lines.length === 0 || !lines[0].includes('WEBVTT')) {
|
||
throw new Error('无效的VTT文件: 缺少WEBVTT文件头');
|
||
}
|
||
// 解析VTT内容
|
||
for (var i = 1; i < lines.length; i++) {
|
||
var line = lines[i].trim();
|
||
|
||
// 跳过空行
|
||
if (line === '') continue;
|
||
|
||
// 检查是否是时间轴
|
||
if (line.includes('-->')) {
|
||
// 如果已有当前cue,先保存它
|
||
if (currentCue) {
|
||
cues.push(currentCue);
|
||
}
|
||
|
||
// 创建新的cue
|
||
currentCue = {
|
||
identifier: null,
|
||
start: null,
|
||
end: null,
|
||
text: '',
|
||
styles: null
|
||
};
|
||
|
||
// 解析时间轴
|
||
var arrowIndex = line.indexOf('-->');
|
||
currentCue.startTime = line.substring(0, arrowIndex).trim();
|
||
currentCue.endTime = line.substring(arrowIndex + 3, arrowIndex + 15).trim();
|
||
var settingsLine = line.substring(arrowIndex + 15).trim();
|
||
// 解析设置(如果有)
|
||
if (settingsLine) currentCue.styles = parseCueSettings(settingsLine);
|
||
}
|
||
// 检查是否是标识符(数字或文本)
|
||
else if (!currentCue && /^\d+$/.test(line)) {
|
||
// 这是标识符行
|
||
if (currentCue) {
|
||
currentCue.identifier = line;
|
||
}
|
||
}
|
||
// 检查是否是注释
|
||
else if (line.startsWith('NOTE')) {
|
||
note = line.substring(4).trim();
|
||
}
|
||
// 检查是否是样式块
|
||
else if (line.startsWith('STYLE')) {
|
||
style = line.substring(5).trim();
|
||
}
|
||
// 检查是否是区域块
|
||
else if (line.startsWith('REGION')) {
|
||
region = line.substring(6).trim();
|
||
}
|
||
// 否则是文本行
|
||
else if (currentCue) {
|
||
if (currentCue.text) {
|
||
currentCue.text += '\n' + line;
|
||
} else {
|
||
currentCue.text = line;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 添加最后一个cue
|
||
if (currentCue) {
|
||
cues.push(currentCue);
|
||
}
|
||
return {
|
||
metadata: {
|
||
note: note,
|
||
style: style,
|
||
region: region
|
||
},
|
||
cues: cues
|
||
};
|
||
}
|
||
}]);
|
||
}(); // 解析脚本信息
|
||
_defineProperty(YbSubtitle, "DEFAULT_CONFIG", {
|
||
color: '#ffffff',
|
||
//文字颜色
|
||
fontSize: 18,
|
||
//字体大小
|
||
bottom: '10%',
|
||
//底部边距
|
||
shadowColor: 'rgba(0,0,0,.5)' //阴影颜色
|
||
});
|
||
function parseScriptInfo(line, infoObj) {
|
||
var colonIndex = line.indexOf(':');
|
||
if (colonIndex === -1) return;
|
||
var key = line.substring(0, colonIndex).trim();
|
||
var value = line.substring(colonIndex + 1).trim();
|
||
infoObj[key] = value;
|
||
}
|
||
|
||
// 解析样式
|
||
function parseStyles(line, stylesObj) {
|
||
if (line.toLowerCase().startsWith('format:')) {
|
||
// 解析格式行
|
||
var formatLine = line.substring(7).trim();
|
||
stylesObj.format = formatLine.split(',').map(function (item) {
|
||
return item.trim();
|
||
});
|
||
} else if (line.toLowerCase().startsWith('style:')) {
|
||
// 解析样式数据
|
||
var styleLine = line.substring(6).trim();
|
||
var values = parseCsvLine(styleLine);
|
||
if (values.length === stylesObj.format.length) {
|
||
var style = {};
|
||
stylesObj.format.forEach(function (key, index) {
|
||
style[key] = values[index];
|
||
});
|
||
stylesObj.data.push(style);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 解析事件
|
||
function parseEvents(line, eventsObj) {
|
||
if (line.toLowerCase().startsWith('format:')) {
|
||
// 解析格式行
|
||
var formatLine = line.substring(7).trim();
|
||
eventsObj.format = formatLine.split(',').map(function (item) {
|
||
return item.trim();
|
||
});
|
||
} else if (line.toLowerCase().startsWith('dialogue:') || line.toLowerCase().startsWith('comment:')) {
|
||
// 解析对话或注释数据
|
||
var typeEnd = line.indexOf(':');
|
||
var type = line.substring(0, typeEnd).trim();
|
||
var dataLine = line.substring(typeEnd + 1).trim();
|
||
var values = parseCsvLine(dataLine);
|
||
if (values.length === eventsObj.format.length) {
|
||
var event = {
|
||
Type: type
|
||
};
|
||
eventsObj.format.forEach(function (key, index) {
|
||
event[key] = values[index];
|
||
});
|
||
eventsObj.data.push(event);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 解析CSV格式的行(处理逗号在引号内的情况)
|
||
function parseCsvLine(line) {
|
||
var result = [];
|
||
var current = '';
|
||
var inQuotes = false;
|
||
for (var i = 0; i < line.length; i++) {
|
||
var _char = line[i];
|
||
if (_char === '"') {
|
||
inQuotes = !inQuotes;
|
||
} else if (_char === ',' && !inQuotes) {
|
||
result.push(current.trim());
|
||
current = '';
|
||
} else {
|
||
current += _char;
|
||
}
|
||
}
|
||
result.push(current.trim());
|
||
return result;
|
||
}
|
||
|
||
// 解析Cue设置
|
||
function parseCueSettings(settingsLine) {
|
||
var settings = {};
|
||
var parts = settingsLine.split(' ');
|
||
var _iterator = _createForOfIteratorHelper(parts),
|
||
_step;
|
||
try {
|
||
for (_iterator.s(); !(_step = _iterator.n()).done;) {
|
||
var part = _step.value;
|
||
var _part$split = part.split(':'),
|
||
_part$split2 = _slicedToArray(_part$split, 2),
|
||
key = _part$split2[0],
|
||
value = _part$split2[1];
|
||
if (key && value) {
|
||
settings[key] = value;
|
||
}
|
||
}
|
||
} catch (err) {
|
||
_iterator.e(err);
|
||
} finally {
|
||
_iterator.f();
|
||
}
|
||
return settings;
|
||
}
|
||
//兼容new Function,为了挂载到window对象上
|
||
if (typeof window != 'undefined') {
|
||
window.YbSubtitle = YbSubtitle;
|
||
} |