The Dark Old Days of TTML
How to caption the web using TTML and why you might not want to
Brightcove
Video.js
@gkatsev
http://slides.com/gkatsev/ttml/
Gary Katsevman
TTML
Timed Text Markup Language
Originally, DFXP
Distribution Format Exchange Profile
Standardized by W3C
Related Standards
WebVTT
SMPTE-EE
CFF-TT
Common File Format
Uses SMPTE-TT in UltraViolet
SMPTE-TT?
Society of Motion Picture and Television Engineers
Standards, Recommended Practices, Engineering Guidelines
-
TV Production
-
Filmmaking
-
Digital cinema
-
Audio Recording
-
Information Technology
-
Medical Imaging
Superset of TTML
Maintaining Legacy
CEA-706
CEA-608
DVB
FCC's CVAA
21st Century Communications and Video Accessibility Act
Safe Harbor for SMPTE-TT
2012
WHATWG chose WebVTT in 2010
TTML
Industry Standard
Ingestion Systems
-
YouTube
-
Netflix
-
Amazon
Google Cast
Internet Explorer
Microsoft Edge
Like Html, CSS
But worse
Much worse
<tt xml:lang="en"
xmlns="http://www.w3.org/ns/ttml"
xmlns:tts="http://www.w3.org/ns/ttml#styling">
<head>
<metadata xmlns:ttm="http://www.w3.org/ns/ttml#metadata">
<ttm:title>TTML Example</ttm:title>
<ttm:copyright>Thierry Michel 2015</ttm:copyright>
</metadata>
<styling xmlns:tts="http://www.w3.org/ns/ttml#styling">
<style xml:id="s1" tts:color="red" tts:textAlign="center" />
</styling>
</head>
<body>
<div>
<p xml:id="c1" begin="00:00:00" end="00:00:10">
Hello I am your first line.</p>
<p xml:id="c2" begin="00:00:02" end="00:00:10">
I am your second captions<br/>but with two lines.</p>
<p xml:id="c3" xml:lang="fr" begin="00:00:04" end="00:00:10">
Je suis le troisième sous-titre.</p>
<p xml:id="c4" begin="00:00:06" end="00:00:10" >
I am another caption with <span tts:fontWeight="bold">Bold</span> and <span tts:fontStyle="italic">Italic</span> styles.</p>
<p xml:id="c5" begin="00:00:08" end="00:00:10" style="s1">
I am the last caption displayed in red and centered.</p>
</div>
</body>
</tt>
TTML
WEBVTT
title: Web-VTT Example
copyright: Thierry Michel 2015
NOTE to style this document, CSS color font must be the specified in the HTML hosting page in the <style> element.
c1
00:00.000 --> 00:10.000
Hello I am your first line.
c2
00:02.000 --> 00:10.000
I am your second captions
but with two lines.
c3
00:04.000 --> 00:10.000
<lang fr>Je suis le troisième sous-titre.</lang>
c4
00:06.000 --> 00:10.000
I am another caption with <b>Bold</b> and <i>Italic</i> styles.
c5
00:08.000 --> 00:10.000 align:middle
<c.red>I am the last caption displayed in red and centered.</c>
--------------------------------------
In the HTML 5, CSS style:
<style> ::cue(.red) {color:red;} </style>
VTT
<tt xml:lang="en"
xmlns="http://www.w3.org/ns/ttml"
xmlns:tts="http://www.w3.org/ns/ttml#styling">
<head>
<metadata xmlns:ttm="http://www.w3.org/ns/ttml#metadata">
<ttm:title>TTML Example</ttm:title>
<ttm:copyright>Thierry Michel 2015</ttm:copyright>
</metadata>
<styling xmlns:tts="http://www.w3.org/ns/ttml#styling">
<style xml:id="s1" tts:color="red" tts:textAlign="center"/>
</styling>
</head>
<body>
<div>
<p xml:id="c1" begin="00:00:00" end="00:00:10">
Hello I am your first line.</p>
<p xml:id="c2" begin="00:00:02" end="00:00:10">
I am your second captions<br/>but with two lines.</p>
</div>
</body>
<body>
<div>
<p xml:id="c3" xml:lang="fr" begin="00:00:04" end="00:00:10">
Je suis le troisième sous-titre.</p>
<p xml:id="c4" begin="00:00:06" end="00:00:10" >
I am another caption with
<span tts:fontWeight="bold">Bold</span> and
<span tts:fontStyle="italic">Italic</span> styles.</p>
</div>
</body>
<head>
<styling xmlns:tts="http://www.w3.org/ns/ttml#styling">
<style xml:id="s1" tts:color="red" tts:textAlign="center" />
</styling>
</head>
<body>
<div>
<p xml:id="c5" begin="00:00:08" end="00:00:10" style="s1">
I am the last caption displayed in red and centered.</p>
</div>
</body>
videojs-ttml
player.ttml({
src: 'test/fixtures/full.xml',
label: 'English',
srclang: 'en-us'
});
videojs.plugin('ttml', function(options) {
const player = this;
let ttmlDisplay;
const dfxpurl = options.src;
videojs.xhr({
uri: dfxpurl
}, function(err, res, body) {
if (res.statusCode >= 400) {
return videojs.error(err);
}
const styleParser = parseTtml(body, player.currentHeight());
player.ttml.styleParser = styleParser;
ttmlDisplay = player.addChild('ttmlDisplay');
});
document.querySelector('.on-button')
.addEventListener('click', function() {
ttmlDisplay.enable();
});
document.querySelector('.off-button')
.addEventListener('click', function() {
ttmlDisplay.disable();
});
document.querySelector('.en-button')
.addEventListener('click', function() {
ttmlDisplay.setLanguage('en');
ttmlDisplay.enable();
});
document.querySelector('.fr-button')
.addEventListener('click', function() {
ttmlDisplay.setLanguage('fr');
ttmlDisplay.enable();
});
const parseTtml = function(ttml, height) {
const dom = (new DOMParser())
.parseFromString(ttml, 'text/xml');
if (!dom || dom.querySelector('parsererror')) {
videojs.error('failed to parse the TTML');
}
// remove all potential <script> elements
const domScripts = dom.querySelectorAll('script');
for (let i = 0; i < domScripts; i++) {
const domScript = domScripts[i];
domScript.parentNode.removeChild(domScript);
}
const language = dom.documentElement.getAttribute('xml:lang');
const body = dom.querySelector('body');
const styleParser = new TtmlDomParser(dom, {}, height);
return styleParser;
};
const TtmlDomParser = function(dom, styleOptions, height) {
/* various other setup */
this.setNamespaces(dom);
this.parseHead(dom);
this.parseBody(dom);
};
Brightcove
Video.js
@gkatsev
http://slides.com/gkatsev/ttml/
Gary Katsevman
Thanks!
The Dark Old Days of TTML
By Gary Katsevman
The Dark Old Days of TTML
How to caption the web using TTML and why you might not want to
- 649