rx-player
Version:
Canal+ HTML5 Video Player
184 lines (164 loc) • 5.31 kB
text/typescript
/**
* Copyright 2015 CANAL+ Group
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* /!\ This file is feature-switchable.
* It always should be imported through the `features` object.
*/
// Parse SRT subtitles into HTML.
// Done for fun. Understand <b>, <i>, <u> and <font color="#ff0000" /> type
// of tags.
import getCueBlocks from "./get_cue_blocks";
import parseCueBlock from "./parse_cue";
export interface ISRTHTMLCue {
start: number;
end: number;
element: HTMLElement;
}
/**
* @param {string} srtStr
* @param {Number} timeOffset
* @returns {Array.<Object>}
*/
export default function parseSRTStringToHTML(
srtStr: string,
timeOffset: number,
): ISRTHTMLCue[] {
// Even if srt only authorize CRLF, we will also take LF or CR as line
// terminators for resilience
const lines = srtStr.split(/\r\n|\n|\r/);
const cueBlocks: string[][] = getCueBlocks(lines);
const cues: ISRTHTMLCue[] = [];
for (const cueBlock of cueBlocks) {
const cueObject = parseCueBlock(cueBlock, timeOffset);
if (cueObject !== null) {
const htmlCue = toHTML(cueObject);
if (htmlCue !== null) {
cues.push(htmlCue);
}
}
}
return cues;
}
/**
* @param {Object} cueObj
* @param {number} cueObj.start
* @param {number} cueObj.end
* @param {Array.<string>} cueObj.payload
* @returns {Object|null}
*/
function toHTML(cueObj: {
start: number;
end: number;
payload: string[];
}): ISRTHTMLCue | null {
const { start, end, payload } = cueObj;
const pEl = document.createElement("div");
pEl.className = "rxp-texttrack-p";
pEl.style.fontSize = "28px";
pEl.style.position = "absolute";
pEl.style.bottom = "5%";
pEl.style.width = "100%";
pEl.style.textAlign = "center";
pEl.style.color = "#fff";
pEl.style.textShadow =
"-1px -1px 2px #000," +
"1px -1px 2px #000," +
"-1px 1px 2px #000," +
"1px 1px 2px #000";
for (let i = 0; i < payload.length; i++) {
if (i !== 0) {
pEl.appendChild(document.createElement("br"));
}
const span = generateSpansFromSRTText(payload[i]);
pEl.appendChild(span);
}
return {
start,
end,
element: pEl,
};
}
/**
* Take a single srt line and convert it into a span with the right style while
* avoiding XSS.
* What we do is set a whitelist of authorized tags, and recreate the
* corresponding tag from scratch.
* Supported tags:
* - <b>: make content bold
* - <i>: make content italic
* - <u>: draw underline on content
* - <font color="x">: add color x to the content
* @param {string} text
* @returns {HTMLElement}
*/
function generateSpansFromSRTText(text: string): HTMLElement {
const secureDiv = document.createElement("div");
secureDiv.innerHTML = text;
const _loop = function (node: Node): HTMLElement {
const childNodes = node.childNodes;
const span = document.createElement("span");
span.className = "rxp-texttrack-span";
for (let i = 0; i < childNodes.length; i++) {
const currentNode = childNodes[i];
if (currentNode.nodeName === "#text") {
const linifiedText = (currentNode as Text).wholeText.split("\n");
for (let line = 0; line < linifiedText.length; line++) {
if (line !== 0) {
span.appendChild(document.createElement("br"));
}
if (linifiedText[line].length > 0) {
const textNode = document.createTextNode(linifiedText[line]);
span.appendChild(textNode);
}
}
} else if (currentNode.nodeName === "B") {
const spanChild = _loop(currentNode);
spanChild.style.fontWeight = "bold";
span.appendChild(spanChild);
} else if (currentNode.nodeName === "I") {
const spanChild = _loop(currentNode);
spanChild.style.fontStyle = "italic";
span.appendChild(spanChild);
} else if (currentNode.nodeName === "U") {
const spanChild = _loop(currentNode);
spanChild.style.textDecoration = "underline";
span.appendChild(spanChild);
} else if (
isNodeFontWithColorProp(currentNode) &&
typeof currentNode.color === "string"
) {
// TODO loop through attributes to find color?
const spanChild = _loop(currentNode);
spanChild.style.color = currentNode.color;
span.appendChild(spanChild);
} else {
const spanChild = _loop(currentNode);
span.appendChild(spanChild);
}
}
return span;
};
return _loop(secureDiv);
}
/**
* Returns `true` if the given node is a `<font>` element which contains a
* `color` attribute.
* @param {Node} node
* @returns {boolean}
*/
function isNodeFontWithColorProp(node: Node): node is Node & { color: unknown } {
return node.nodeName === "FONT" && "color" in node;
}