2013-05-21 16:14:00 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
#include "mozilla/dom/HTMLTrackElement.h"
|
|
|
|
#include "mozilla/dom/TextTrackCue.h"
|
|
|
|
#include "mozilla/dom/TextTrackCueBinding.h"
|
2013-06-10 15:30:00 +00:00
|
|
|
#include "mozilla/dom/ProcessingInstruction.h"
|
|
|
|
#include "nsIFrame.h"
|
|
|
|
#include "nsTextNode.h"
|
|
|
|
#include "nsVideoFrame.h"
|
2013-05-21 16:14:00 +00:00
|
|
|
#include "webvtt/cue.h"
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace dom {
|
|
|
|
|
2013-06-10 15:30:00 +00:00
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE_4(TextTrackCue,
|
2013-05-21 16:14:00 +00:00
|
|
|
mGlobal,
|
|
|
|
mTrack,
|
2013-06-10 15:30:00 +00:00
|
|
|
mTrackElement,
|
|
|
|
mCueDiv)
|
2013-05-21 16:14:00 +00:00
|
|
|
|
|
|
|
NS_IMPL_ADDREF_INHERITED(TextTrackCue, nsDOMEventTargetHelper)
|
|
|
|
NS_IMPL_RELEASE_INHERITED(TextTrackCue, nsDOMEventTargetHelper)
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(TextTrackCue)
|
|
|
|
NS_INTERFACE_MAP_END_INHERITING(nsDOMEventTargetHelper)
|
|
|
|
|
|
|
|
// Set cue setting defaults based on step 19 & seq.
|
|
|
|
// in http://dev.w3.org/html5/webvtt/#parsing
|
|
|
|
void
|
|
|
|
TextTrackCue::SetDefaultCueSettings()
|
|
|
|
{
|
|
|
|
mPosition = 50;
|
|
|
|
mSize = 100;
|
|
|
|
mPauseOnExit = false;
|
|
|
|
mSnapToLines = true;
|
|
|
|
mLine = WEBVTT_AUTO;
|
|
|
|
mAlign = TextTrackCueAlign::Middle;
|
|
|
|
}
|
|
|
|
|
|
|
|
TextTrackCue::TextTrackCue(nsISupports* aGlobal,
|
|
|
|
double aStartTime,
|
|
|
|
double aEndTime,
|
|
|
|
const nsAString& aText)
|
|
|
|
: mGlobal(aGlobal)
|
|
|
|
, mText(aText)
|
|
|
|
, mStartTime(aStartTime)
|
|
|
|
, mEndTime(aEndTime)
|
|
|
|
, mHead(nullptr)
|
|
|
|
{
|
|
|
|
SetDefaultCueSettings();
|
|
|
|
MOZ_ASSERT(aGlobal);
|
|
|
|
SetIsDOMBinding();
|
|
|
|
}
|
|
|
|
|
|
|
|
TextTrackCue::TextTrackCue(nsISupports* aGlobal,
|
|
|
|
double aStartTime,
|
|
|
|
double aEndTime,
|
|
|
|
const nsAString& aText,
|
|
|
|
HTMLTrackElement* aTrackElement,
|
|
|
|
webvtt_node* head)
|
|
|
|
: mGlobal(aGlobal)
|
|
|
|
, mText(aText)
|
|
|
|
, mStartTime(aStartTime)
|
|
|
|
, mEndTime(aEndTime)
|
|
|
|
, mTrackElement(aTrackElement)
|
|
|
|
, mHead(head)
|
|
|
|
{
|
|
|
|
// Use the webvtt library's reference counting.
|
|
|
|
webvtt_ref_node(mHead);
|
|
|
|
SetDefaultCueSettings();
|
|
|
|
MOZ_ASSERT(aGlobal);
|
|
|
|
SetIsDOMBinding();
|
|
|
|
}
|
|
|
|
|
|
|
|
TextTrackCue::~TextTrackCue()
|
|
|
|
{
|
|
|
|
if (mHead) {
|
|
|
|
// Release our reference and null mHead.
|
|
|
|
webvtt_release_node(&mHead);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2013-06-10 15:30:00 +00:00
|
|
|
TextTrackCue::CreateCueOverlay()
|
2013-05-21 16:14:00 +00:00
|
|
|
{
|
2013-06-19 18:02:04 +00:00
|
|
|
nsCOMPtr<nsPIDOMWindow> window(do_QueryInterface(mGlobal));
|
|
|
|
if(!window) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
nsIDocument* document = window->GetDoc();
|
|
|
|
if(!document) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
document->CreateElem(NS_LITERAL_STRING("div"), nullptr,
|
|
|
|
kNameSpaceID_XHTML,
|
|
|
|
getter_AddRefs(mCueDiv));
|
2013-06-10 15:30:00 +00:00
|
|
|
nsGenericHTMLElement* cueDiv =
|
|
|
|
static_cast<nsGenericHTMLElement*>(mCueDiv.get());
|
|
|
|
cueDiv->SetClassName(NS_LITERAL_STRING("caption-text"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
TextTrackCue::RenderCue()
|
|
|
|
{
|
|
|
|
nsRefPtr<DocumentFragment> frag = GetCueAsHTML();
|
|
|
|
if (!frag || !mTrackElement) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mCueDiv) {
|
|
|
|
CreateCueOverlay();
|
|
|
|
}
|
|
|
|
|
|
|
|
HTMLMediaElement* parent = mTrackElement->mMediaParent;
|
|
|
|
if (!parent) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIFrame* frame = parent->GetPrimaryFrame();
|
|
|
|
if (!frame) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsVideoFrame* videoFrame = do_QueryFrame(frame);
|
|
|
|
if (!videoFrame) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIContent* overlay = videoFrame->GetCaptionOverlay();
|
|
|
|
if (!overlay) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ErrorResult rv;
|
|
|
|
nsContentUtils::SetNodeTextContent(overlay, EmptyString(), true);
|
|
|
|
nsContentUtils::SetNodeTextContent(mCueDiv, EmptyString(), true);
|
|
|
|
|
|
|
|
mCueDiv->AppendChild(*frag, rv);
|
|
|
|
overlay->AppendChild(*mCueDiv, rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<DocumentFragment>
|
|
|
|
TextTrackCue::GetCueAsHTML()
|
|
|
|
{
|
2013-06-19 18:02:04 +00:00
|
|
|
nsCOMPtr<nsPIDOMWindow> window(do_QueryInterface(mGlobal));
|
|
|
|
if(!window) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
nsIDocument* document = window->GetDoc();
|
|
|
|
if(!document){
|
|
|
|
return nullptr;
|
|
|
|
}
|
2013-06-10 15:30:00 +00:00
|
|
|
nsRefPtr<DocumentFragment> frag =
|
2013-06-19 18:02:04 +00:00
|
|
|
document->CreateDocumentFragment();
|
2013-06-10 15:30:00 +00:00
|
|
|
|
|
|
|
ConvertNodeTreeToDOMTree(frag);
|
|
|
|
|
|
|
|
return frag.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
struct WebVTTNodeParentPair
|
|
|
|
{
|
|
|
|
webvtt_node* mNode;
|
|
|
|
nsIContent* mParent;
|
|
|
|
|
|
|
|
WebVTTNodeParentPair(webvtt_node* aNode, nsIContent* aParent)
|
|
|
|
: mNode(aNode)
|
|
|
|
, mParent(aParent)
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
PushChildren(nsTArray<WebVTTNodeParentPair> &aNodeParentPairStack,
|
|
|
|
webvtt_node* aNode, nsIContent* aParentContent)
|
|
|
|
{
|
|
|
|
// Push on in reverse order so we process the nodes in the correct
|
|
|
|
// order -- left to right.
|
|
|
|
for (int i = aNode->data.internal_data->length; i > 0; i--) {
|
|
|
|
WebVTTNodeParentPair nodeParentPair(
|
|
|
|
aNode->data.internal_data->children[i - 1],
|
|
|
|
aParentContent);
|
|
|
|
aNodeParentPairStack.AppendElement(nodeParentPair);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static WebVTTNodeParentPair
|
|
|
|
PopChild(nsTArray<WebVTTNodeParentPair> &aNodeParentPairStack) {
|
|
|
|
WebVTTNodeParentPair temp =
|
|
|
|
aNodeParentPairStack.LastElement();
|
|
|
|
aNodeParentPairStack.RemoveElementAt(aNodeParentPairStack.Length() - 1);
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
TextTrackCue::ConvertNodeTreeToDOMTree(nsIContent* aParentContent)
|
|
|
|
{
|
|
|
|
nsTArray<WebVTTNodeParentPair> nodeParentPairStack;
|
|
|
|
|
|
|
|
// mHead should actually be the head of a node tree.
|
2013-06-19 18:02:04 +00:00
|
|
|
if (!mHead || mHead->kind != WEBVTT_HEAD_NODE) {
|
2013-06-10 15:30:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Seed the stack for traversal.
|
|
|
|
PushChildren(nodeParentPairStack, mHead, aParentContent);
|
|
|
|
|
|
|
|
while (!nodeParentPairStack.IsEmpty()) {
|
|
|
|
WebVTTNodeParentPair nodeParentPair = PopChild(nodeParentPairStack);
|
|
|
|
nsCOMPtr<nsIContent> content;
|
|
|
|
if (WEBVTT_IS_VALID_LEAF_NODE(nodeParentPair.mNode->kind)) {
|
|
|
|
content = ConvertLeafNodeToContent(nodeParentPair.mNode);
|
|
|
|
} else if (WEBVTT_IS_VALID_INTERNAL_NODE(nodeParentPair.mNode->kind)) {
|
|
|
|
content = ConvertInternalNodeToContent(nodeParentPair.mNode);
|
|
|
|
// Push the children of the current node onto the stack for traversal.
|
|
|
|
PushChildren(nodeParentPairStack, nodeParentPair.mNode, content);
|
|
|
|
}
|
|
|
|
if (content && nodeParentPair.mParent) {
|
|
|
|
ErrorResult rv;
|
|
|
|
nodeParentPair.mParent->AppendChild(*content, rv);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<nsIContent>
|
|
|
|
TextTrackCue::ConvertInternalNodeToContent(const webvtt_node* aWebVTTNode)
|
|
|
|
{
|
|
|
|
nsIAtom* atom;
|
|
|
|
|
|
|
|
switch (aWebVTTNode->kind) {
|
|
|
|
case WEBVTT_BOLD:
|
|
|
|
atom = nsGkAtoms::b;
|
|
|
|
break;
|
|
|
|
case WEBVTT_ITALIC:
|
|
|
|
atom = nsGkAtoms::i;
|
|
|
|
break;
|
|
|
|
case WEBVTT_UNDERLINE:
|
|
|
|
atom = nsGkAtoms::u;
|
|
|
|
break;
|
|
|
|
case WEBVTT_RUBY:
|
|
|
|
atom = nsGkAtoms::ruby;
|
|
|
|
break;
|
|
|
|
case WEBVTT_RUBY_TEXT:
|
|
|
|
atom = nsGkAtoms::rt;
|
|
|
|
break;
|
|
|
|
case WEBVTT_VOICE:
|
|
|
|
atom = nsGkAtoms::span;
|
|
|
|
break;
|
|
|
|
case WEBVTT_CLASS:
|
|
|
|
atom = nsGkAtoms::span;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return nullptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsIContent> cueTextContent;
|
2013-06-19 18:02:04 +00:00
|
|
|
nsCOMPtr<nsPIDOMWindow> window(do_QueryInterface(mGlobal));
|
|
|
|
if(!window) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
nsIDocument* document = window->GetDoc();
|
|
|
|
if(!document){
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
document->CreateElem(nsDependentAtomString(atom), nullptr,
|
|
|
|
kNameSpaceID_XHTML,
|
|
|
|
getter_AddRefs(cueTextContent));
|
2013-06-10 15:30:00 +00:00
|
|
|
|
|
|
|
if (aWebVTTNode->kind == WEBVTT_VOICE) {
|
|
|
|
const char* text =
|
|
|
|
webvtt_string_text(&aWebVTTNode->data.internal_data->annotation);
|
|
|
|
if (text) {
|
|
|
|
nsGenericHTMLElement* genericHtmlElement =
|
|
|
|
static_cast<nsGenericHTMLElement*>(cueTextContent.get());
|
|
|
|
genericHtmlElement->SetTitle(NS_ConvertUTF8toUTF16(text));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
webvtt_stringlist* classes = aWebVTTNode->data.internal_data->css_classes;
|
|
|
|
if (classes && classes->items && classes->length > 0) {
|
|
|
|
nsAutoString classString;
|
|
|
|
|
|
|
|
const char *text = webvtt_string_text(classes->items);
|
|
|
|
if (text) {
|
|
|
|
AppendUTF8toUTF16(text, classString);
|
|
|
|
for (uint32_t i = 1; i < classes->length; i++) {
|
|
|
|
text = webvtt_string_text(classes->items + i);
|
|
|
|
if (text) {
|
|
|
|
classString.Append(' ');
|
|
|
|
AppendUTF8toUTF16(text, classString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nsGenericHTMLElement* genericHtmlElement =
|
|
|
|
static_cast<nsGenericHTMLElement*>(cueTextContent.get());
|
|
|
|
genericHtmlElement->SetClassName(classString);
|
|
|
|
}
|
|
|
|
return cueTextContent.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<nsIContent>
|
|
|
|
TextTrackCue::ConvertLeafNodeToContent(const webvtt_node* aWebVTTNode)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIContent> cueTextContent;
|
2013-06-19 18:02:04 +00:00
|
|
|
nsCOMPtr<nsPIDOMWindow> window(do_QueryInterface(mGlobal));
|
|
|
|
if(!window) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
nsIDocument* document = window->GetDoc();
|
|
|
|
if(!document) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
nsNodeInfoManager* nimgr = document->NodeInfoManager();
|
2013-06-10 15:30:00 +00:00
|
|
|
switch (aWebVTTNode->kind) {
|
|
|
|
case WEBVTT_TEXT:
|
|
|
|
{
|
|
|
|
cueTextContent = new nsTextNode(nimgr);
|
|
|
|
const char* text = webvtt_string_text(&aWebVTTNode->data.text);
|
|
|
|
if (text) {
|
|
|
|
cueTextContent->SetText(NS_ConvertUTF8toUTF16(text), false);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case WEBVTT_TIME_STAMP:
|
|
|
|
{
|
|
|
|
nsAutoString timeStamp;
|
|
|
|
timeStamp.AppendInt(aWebVTTNode->data.timestamp);
|
|
|
|
cueTextContent =
|
|
|
|
NS_NewXMLProcessingInstruction(nimgr, NS_LITERAL_STRING("timestamp"),
|
|
|
|
timeStamp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return nullptr;
|
|
|
|
break;
|
2013-05-21 16:14:00 +00:00
|
|
|
}
|
2013-06-10 15:30:00 +00:00
|
|
|
return cueTextContent.forget();
|
2013-05-21 16:14:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
JSObject*
|
|
|
|
TextTrackCue::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aScope)
|
|
|
|
{
|
|
|
|
return TextTrackCueBinding::Wrap(aCx, aScope, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
TextTrackCue::CueChanged()
|
|
|
|
{
|
|
|
|
if (mTrack) {
|
|
|
|
mTrack->CueChanged(*this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} // namespace dom
|
|
|
|
} // namespace mozilla
|