/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim:set ts=2 sw=2 et tw=78: */ /* 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 "MediaTrack.h" #include "MediaTrackList.h" #include "mozilla/AsyncEventDispatcher.h" #include "mozilla/dom/HTMLMediaElement.h" #include "mozilla/dom/AudioTrack.h" #include "mozilla/dom/VideoTrack.h" #include "mozilla/dom/TrackEvent.h" #include "nsThreadUtils.h" namespace mozilla { namespace dom { void MediaTrackListListener::NotifyMediaTrackCreated(MediaTrack* aTrack) { if (!mMediaTrackList && !aTrack) { return; } if (aTrack->AsAudioTrack() && mMediaTrackList->AsAudioTrackList()) { mMediaTrackList->AddTrack(aTrack); } else if (aTrack->AsVideoTrack() && mMediaTrackList->AsVideoTrackList()) { mMediaTrackList->AddTrack(aTrack); } } void MediaTrackListListener::NotifyMediaTrackEnded(const nsAString& aId) { if (!mMediaTrackList) { return; } const nsRefPtr track = mMediaTrackList->GetTrackById(aId); if (track) { mMediaTrackList->RemoveTrack(track); } } MediaTrackList::MediaTrackList(nsPIDOMWindow* aOwnerWindow, HTMLMediaElement* aMediaElement) : DOMEventTargetHelper(aOwnerWindow) , mMediaElement(aMediaElement) { } MediaTrackList::~MediaTrackList() { } NS_IMPL_CYCLE_COLLECTION_INHERITED(MediaTrackList, DOMEventTargetHelper, mTracks, mMediaElement) NS_IMPL_ADDREF_INHERITED(MediaTrackList, DOMEventTargetHelper) NS_IMPL_RELEASE_INHERITED(MediaTrackList, DOMEventTargetHelper) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(MediaTrackList) NS_INTERFACE_MAP_END_INHERITING(DOMEventTargetHelper) MediaTrack* MediaTrackList::operator[](uint32_t aIndex) { return mTracks.ElementAt(aIndex); } MediaTrack* MediaTrackList::IndexedGetter(uint32_t aIndex, bool& aFound) { aFound = aIndex < mTracks.Length(); return aFound ? mTracks[aIndex] : nullptr; } MediaTrack* MediaTrackList::GetTrackById(const nsAString& aId) { for (uint32_t i = 0; i < mTracks.Length(); ++i) { if (aId.Equals(mTracks[i]->GetId())) { return mTracks[i]; } } return nullptr; } void MediaTrackList::AddTrack(MediaTrack* aTrack) { mTracks.AppendElement(aTrack); aTrack->Init(GetOwner()); aTrack->SetTrackList(this); CreateAndDispatchTrackEventRunner(aTrack, NS_LITERAL_STRING("addtrack")); } void MediaTrackList::RemoveTrack(const nsRefPtr& aTrack) { mTracks.RemoveElement(aTrack); aTrack->SetTrackList(nullptr); CreateAndDispatchTrackEventRunner(aTrack, NS_LITERAL_STRING("removetrack")); } already_AddRefed MediaTrackList::CreateAudioTrack(const nsAString& aId, const nsAString& aKind, const nsAString& aLabel, const nsAString& aLanguage, bool aEnabled) { nsRefPtr track = new AudioTrack(aId, aKind, aLabel, aLanguage, aEnabled); return track.forget(); } already_AddRefed MediaTrackList::CreateVideoTrack(const nsAString& aId, const nsAString& aKind, const nsAString& aLabel, const nsAString& aLanguage) { nsRefPtr track = new VideoTrack(aId, aKind, aLabel, aLanguage); return track.forget(); } void MediaTrackList::EmptyTracks() { for (uint32_t i = 0; i < mTracks.Length(); ++i) { mTracks[i]->SetTrackList(nullptr); } mTracks.Clear(); } void MediaTrackList::CreateAndDispatchChangeEvent() { nsRefPtr asyncDispatcher = new AsyncEventDispatcher(this, NS_LITERAL_STRING("change"), false); asyncDispatcher->PostDOMEvent(); } void MediaTrackList::CreateAndDispatchTrackEventRunner(MediaTrack* aTrack, const nsAString& aEventName) { TrackEventInit eventInit; if (aTrack->AsAudioTrack()) { eventInit.mTrack.SetValue().SetAsAudioTrack() = aTrack->AsAudioTrack(); } else if (aTrack->AsVideoTrack()) { eventInit.mTrack.SetValue().SetAsVideoTrack() = aTrack->AsVideoTrack(); } nsRefPtr event = TrackEvent::Constructor(this, aEventName, eventInit); nsRefPtr asyncDispatcher = new AsyncEventDispatcher(this, event); asyncDispatcher->PostDOMEvent(); } } // namespace dom } // namespace mozilla