/* -*- 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/TextTrackList.h" #include "mozilla/dom/TextTrackListBinding.h" #include "mozilla/dom/TrackEvent.h" #include "nsThreadUtils.h" #include "mozilla/dom/TextTrackCue.h" namespace mozilla { namespace dom { NS_IMPL_CYCLE_COLLECTION_INHERITED_2(TextTrackList, nsDOMEventTargetHelper, mGlobal, mTextTracks) NS_IMPL_ADDREF_INHERITED(TextTrackList, nsDOMEventTargetHelper) NS_IMPL_RELEASE_INHERITED(TextTrackList, nsDOMEventTargetHelper) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(TextTrackList) NS_INTERFACE_MAP_END_INHERITING(nsDOMEventTargetHelper) TextTrackList::TextTrackList(nsISupports* aGlobal) : mGlobal(aGlobal) { SetIsDOMBinding(); } void TextTrackList::GetAllActiveCues(nsTArray >& aCues) { nsTArray< nsRefPtr > cues; for (uint32_t i = 0; i < Length(); i++) { if (mTextTracks[i]->Mode() != TextTrackMode::Disabled) { mTextTracks[i]->GetActiveCueArray(cues); aCues.AppendElements(cues); } } } JSObject* TextTrackList::WrapObject(JSContext* aCx, JS::Handle aScope) { return TextTrackListBinding::Wrap(aCx, aScope, this); } TextTrack* TextTrackList::IndexedGetter(uint32_t aIndex, bool& aFound) { aFound = aIndex < mTextTracks.Length(); return aFound ? mTextTracks[aIndex] : nullptr; } already_AddRefed TextTrackList::AddTextTrack(HTMLMediaElement* aMediaElement, TextTrackKind aKind, const nsAString& aLabel, const nsAString& aLanguage) { nsRefPtr track = new TextTrack(mGlobal, aMediaElement, aKind, aLabel, aLanguage); if (mTextTracks.AppendElement(track)) { CreateAndDispatchTrackEventRunner(track, NS_LITERAL_STRING("addtrack")); } return track.forget(); } TextTrack* TextTrackList::GetTrackById(const nsAString& aId) { nsAutoString id; for (uint32_t i = 0; i < Length(); i++) { mTextTracks[i]->GetId(id); if (aId.Equals(id)) { return mTextTracks[i]; } } return nullptr; } void TextTrackList::RemoveTextTrack(TextTrack* aTrack) { if (mTextTracks.RemoveElement(aTrack)) { CreateAndDispatchTrackEventRunner(aTrack, NS_LITERAL_STRING("removetrack")); } } void TextTrackList::DidSeek() { for (uint32_t i = 0; i < mTextTracks.Length(); i++) { mTextTracks[i]->SetDirty(); } } class TrackEventRunner MOZ_FINAL: public nsRunnable { public: TrackEventRunner(TextTrackList* aList, nsIDOMEvent* aEvent) : mList(aList) , mEvent(aEvent) {} NS_IMETHOD Run() MOZ_OVERRIDE { return mList->DispatchTrackEvent(mEvent); } private: nsRefPtr mList; nsRefPtr mEvent; }; nsresult TextTrackList::DispatchTrackEvent(nsIDOMEvent* aEvent) { return DispatchTrustedEvent(aEvent); } void TextTrackList::CreateAndDispatchChangeEvent() { nsCOMPtr event; nsresult rv = NS_NewDOMEvent(getter_AddRefs(event), this, nullptr, nullptr); if (NS_FAILED(rv)) { NS_WARNING("Failed to create the error event!"); return; } rv = event->InitEvent(NS_LITERAL_STRING("change"), false, false); if (NS_FAILED(rv)) { NS_WARNING("Failed to init the change event!"); return; } event->SetTrusted(true); nsCOMPtr eventRunner = new TrackEventRunner(this, event); NS_DispatchToMainThread(eventRunner, NS_DISPATCH_NORMAL); } void TextTrackList::CreateAndDispatchTrackEventRunner(TextTrack* aTrack, const nsAString& aEventName) { TrackEventInit eventInit; eventInit.mBubbles = false; eventInit.mCancelable = false; eventInit.mTrack = aTrack; nsRefPtr event = TrackEvent::Constructor(this, aEventName, eventInit); // Dispatch the TrackEvent asynchronously. nsCOMPtr eventRunner = new TrackEventRunner(this, event); NS_DispatchToMainThread(eventRunner, NS_DISPATCH_NORMAL); } } // namespace dom } // namespace mozilla