Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement Overlay events #31

Open
wants to merge 27 commits into
base: development
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
21147c7
[WIP] Add video overlay
jmt-qualabs Nov 7, 2024
c791af1
fix styling of overlay video to match video
ronalduQualabs Nov 19, 2024
aae54fb
Add iframe overlay
jmt-qualabs Nov 19, 2024
b3f25f7
Implement earliestResolutionTime
jmt-qualabs Nov 20, 2024
647c4c4
Change overlay tag
jmt-qualabs Nov 21, 2024
f45c568
Remove overlay at end of event duration
jmt-qualabs Nov 21, 2024
962df58
Implement overlay modes and refId
sebastianpiq Nov 21, 2024
35d8d45
Fix seek overlay
sebastianpiq Nov 21, 2024
f76b901
Implement overlay event z attribute
sebastianpiq Nov 22, 2024
477d070
Add OverlayController
jmt-qualabs Nov 25, 2024
4f10a07
Fix overlay loop
jmt-qualabs Nov 25, 2024
9280240
Implement overlay size and topleft
sebastianpiq Nov 25, 2024
f5f04c3
Implement overlay squeeze content
sebastianpiq Nov 25, 2024
905b24f
Use transitions to manage overlays size
sebastianpiq Nov 26, 2024
ed26429
Improve overlay scheduler
sebastianpiq Nov 27, 2024
418c418
Add missing semicolons
sebastianpiq Nov 27, 2024
121cb0a
Remove unused extend mode event id logic
sebastianpiq Nov 27, 2024
734dc38
Add new squeeze current percentage definition
sebastianpiq Nov 27, 2024
1b97763
Add missing semicolons
sebastianpiq Nov 28, 2024
4151332
Improve overlay select mode logic
sebastianpiq Nov 29, 2024
7973882
Implement unit tests
sebastianpiq Dec 2, 2024
2e69c67
Remove unnecessary it only tests
sebastianpiq Dec 2, 2024
b4dfd2a
Add missing semicolons and fix wrong let declarations
sebastianpiq Dec 2, 2024
cccf004
Fix squeeze current bug
sebastianpiq Dec 5, 2024
29790dd
Add overlay sample page
sebastianpiq Dec 13, 2024
7be71c2
Fix looped overlay
jmt-qualabs Dec 18, 2024
718bb74
Fix overlay controller test
jmt-qualabs Dec 19, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Binary file added samples/overlays/lib/img/dashjs-logo.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
84 changes: 84 additions & 0 deletions samples/overlays/overlays.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
<!DOCTYPE html>
<html>
<head>
<title>Dash.js Rocks</title>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/css/bootstrap.min.css" integrity="sha384-Vkoo8x4CGsO3+Hhxv8T/Q5PaXtkKtu6ug5TOeNV6gBiFeWPGFN9MuhOf23Q9Ifjh" crossorigin="anonymous">
<script src="https://code.jquery.com/jquery-3.4.1.slim.min.js" integrity="sha384-J6qa4849blE2+poT4WnyKhv5vZF5SrPo0iEjwBvKU7imGFAV0wwj1yYfoRSJoZ+n" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/popper.js@1.16.0/dist/umd/popper.min.js" integrity="sha384-Q6E9RHvbIyZFJoft+2mJbHaEWldlvI9IOYy5n3zV9zzTtmI3UksdQRVvoxMfooAo" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/js/bootstrap.min.js" integrity="sha384-wfSDF2E50Y2D1uUdj0O3uMBJnjuUD4Ih7YwaYd1iqfktj0Uod8GCExl3Og8ifwB6" crossorigin="anonymous"></script>
</head>
<body>
<div class="container py-4">
<header class="pb-3 mb-4 border-bottom">
<img src="./lib/img/dashjs-logo.png" width="200">
</header>

<div class="btn-group mb-2">
<button type="button" class="btn btn-primary dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
Select Case
</button>
<div class="dropdown-menu" id="assetDropdown">
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case0.mpd">Case 0: Video</a>
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case1.mpd">Case 1: Iframe</a>
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case2.mpd">Case 2: Preloading</a>
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case3.mpd">Case 3: L shaped</a>
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case4.mpd">Case 4: Looped</a>
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case5.mpd">Case 5: Logo</a>
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case6.mpd">Case 6: Extend</a>
<a class="dropdown-item" href="#" data-url="https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case8.mpd">Case 7: Out bounds</a>
</div>
</div>

<div class="dash-video-player">
<div class="videoContainer" id="videoContainer">
<video id="main-video" preload="auto"></video>
<div id="overlay"></div>
</div>
</div>

<div class="py-2">
<button id="playButton" type="button" class="btn btn-primary">Play/Pause</button>
</div>
</div>

<script src="../../dist/dash.all.debug.js"></script>
<script>
(function () {
const video = document.querySelector("video");
const overlay = document.getElementById("overlay");
const playButton = document.getElementById("playButton");
const assetDropdown = document.getElementById("assetDropdown");

const player = dashjs.MediaPlayer().create();
player.initialize();
player.updateSettings({
debug: {
logLevel: dashjs.Debug.LOG_LEVEL_DEBUG,
},
});
player.setAutoPlay(false);
player.attachView(video);
player.attachOverlayRenderingDiv(overlay);

let currentAsset = "https://comcast-dash-6-assets.s3.us-east-2.amazonaws.com/Overlay/case8-b/case0.mpd";
player.attachSource(currentAsset);

playButton.addEventListener("click", () => {
if (video.paused) {
video.play();
} else {
video.pause();
}
});

assetDropdown.addEventListener("click", (event) => {
const target = event.target;
if (target.tagName === "A" && target.dataset.url) {
currentAsset = target.dataset.url;
player.attachSource(currentAsset);
}
});
})();
</script>
</body>
</html>
1 change: 1 addition & 0 deletions src/dash/constants/DashConstants.js
Original file line number Diff line number Diff line change
Expand Up @@ -122,6 +122,7 @@ export default {
MPD: 'MPD',
ORIGINAL_MPD_ID: 'mpdId',
ORIGINAL_PUBLISH_TIME: 'originalPublishTime',
OVERLAY: 'OverlayEvent',
PATCH_LOCATION: 'PatchLocation',
PERIOD: 'Period',
PRESENTATION_TIME: 'presentationTime',
Expand Down
6 changes: 6 additions & 0 deletions src/dash/models/DashManifestModel.js
Original file line number Diff line number Diff line change
Expand Up @@ -1061,6 +1061,12 @@ function DashManifestModel() {
} else {
event.id = null;
}
if (currentMpdEvent.hasOwnProperty(DashConstants.OVERLAY)) {
event.overlay = currentMpdEvent.OverlayEvent;
if (event.overlay.earliestResolutionTime) {
event.calculatedPresentationTime -= event.overlay.earliestResolutionTime / eventStream.timescale;
}
}

if (currentMpdEvent.Signal && currentMpdEvent.Signal.Binary) {
// toString is used to manage both regular and namespaced tags
Expand Down
46 changes: 46 additions & 0 deletions src/streaming/MediaPlayer.js
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ import CatchupController from './controllers/CatchupController.js';
import ServiceDescriptionController from '../dash/controllers/ServiceDescriptionController.js';
import ContentSteeringController from '../dash/controllers/ContentSteeringController.js';
import MediaController from './controllers/MediaController.js';
import OverlayController from './controllers/OverlayController.js';
import BaseURLController from './controllers/BaseURLController.js';
import ManifestLoader from './ManifestLoader.js';
import ErrorHandler from './utils/ErrorHandler.js';
Expand Down Expand Up @@ -144,6 +145,7 @@ function MediaPlayer() {
schemeLoaderFactory,
timelineConverter,
mediaController,
overlayController,
protectionController,
metricsReportingController,
mssHandler,
Expand Down Expand Up @@ -257,6 +259,9 @@ function MediaPlayer() {
if (config.mediaController) {
mediaController = config.mediaController;
}
if (config.overlayController) {
overlayController = config.overlayController;
}
if (config.settings) {
settings = config.settings;
}
Expand Down Expand Up @@ -328,6 +333,10 @@ function MediaPlayer() {
mediaController = MediaController(context).getInstance();
}

if (!overlayController) {
overlayController = OverlayController(context).getInstance();
}

if (!streamController) {
streamController = StreamController(context).getInstance();
}
Expand Down Expand Up @@ -413,6 +422,10 @@ function MediaPlayer() {
videoModel
});

overlayController.setConfig({
sebastianpiq marked this conversation as resolved.
Show resolved Hide resolved
videoModel
});

mediaPlayerModel.setConfig({
playbackController,
serviceDescriptionController
Expand Down Expand Up @@ -472,6 +485,9 @@ function MediaPlayer() {
if (customParametersModel) {
customParametersModel.reset();
}
if (overlayController) {
overlayController.reset()
}

settings.reset();

Expand Down Expand Up @@ -1468,6 +1484,16 @@ function MediaPlayer() {
return videoModel ? videoModel.getTTMLRenderingDiv() : null;
}

/**
* Returns instance of Div that was attached by calling attachOverlayRenderingDiv()
* @returns {Object}
* @memberof module:MediaPlayer
* @instance
*/
function getOverlayRenderingDiv() {
return videoModel ? videoModel.getOverlayRenderingDiv() : null;
}

/**
* Use this method to attach an HTML5 div for dash.js to render rich TTML subtitles.
*
Expand All @@ -1490,6 +1516,18 @@ function MediaPlayer() {
videoModel.setVttRenderingDiv(div);
}

function attachOverlayRenderingDiv(overlayDiv) {
const videoElement = videoModel.getElement();
if (!videoElement) {
throw ELEMENT_NOT_ATTACHED_ERROR;
}
N1Knight marked this conversation as resolved.
Show resolved Hide resolved

overlayController.configureVideoElementForOverlay();
videoModel.setOverlayRenderingDiv(overlayDiv);

overlayController.setupOverlayEvents();
}

/*
---------------------------------------------------------------------------

Expand Down Expand Up @@ -2088,6 +2126,12 @@ function MediaPlayer() {
_resetPlaybackControllers();
}

const overlayDiv = videoModel.getOverlayRenderingDiv()

if (overlayDiv) {
overlayController.reset();
}

if (isReady()) {
_initializePlayback(providedStartTime);
}
Expand Down Expand Up @@ -2740,6 +2784,7 @@ function MediaPlayer() {
attachProtectionController,
attachSource,
attachTTMLRenderingDiv,
attachOverlayRenderingDiv,
attachView,
attachVttRenderingDiv,
clearDefaultUTCTimingSources,
Expand Down Expand Up @@ -2778,6 +2823,7 @@ function MediaPlayer() {
getSource,
getStreamsFromManifest,
getTTMLRenderingDiv,
getOverlayRenderingDiv,
getTargetLiveDelay,
getTracksFor,
getTracksForTypeFromManifest,
Expand Down
9 changes: 9 additions & 0 deletions src/streaming/constants/Constants.js
Original file line number Diff line number Diff line change
Expand Up @@ -341,5 +341,14 @@ export default {
DTSC: 'dtsc',
AVC: 'avc',
HEVC: 'hevc'
},

OVERLAY: {
SCHEME_ID: 'urn:scte:dash:scte214-events',
START_MODE: 'start',
STOP_MODE: 'stop',
EXTEND_MODE: 'extend',
VIDEO_MIMETYPE: 'video/mp4',
IFRMAE_MIMETYPE: 'text/html'
}
}
Loading
Loading