add Ox.VideoPlayer, demos/video/, and prove Chrome doesn't get buffered time ranges right

This commit is contained in:
rolux 2011-05-12 12:39:48 +02:00
parent 15257623f3
commit 13b887abfb
5 changed files with 364 additions and 2 deletions

10
demos/video/index.html Normal file
View file

@ -0,0 +1,10 @@
<!DOCTYPE HTML>
<html>
<head>
<title>OxJS Video Demo</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
<script type="text/javascript" src="../../build/Ox.js"></script>
<script type="text/javascript" src="js/video.js"></script>
</head>
<body></body>
</html>

17
demos/video/js/video.js Normal file
View file

@ -0,0 +1,17 @@
Ox.load('UI', {
debug: true,
theme: 'modern'
}, function() {
var id = '0393109';
var url = 'http://next.0xdb.org/' + id + '/96p.webm?' + Ox.random(1000000);
var timeline = 'http://next.0xdb.org/' + id + '/timeline.16.png';
Ox.UI.$body.css({
padding: '16px'
});
Ox.VideoPlayer({
height: 288,
timeline: timeline,
url: url,
width: 540
}).appendTo(Ox.UI.$body);
});

View file

@ -331,7 +331,7 @@ Ox.Element = function() {
Ox.forEach(Ox.makeObject(arguments), function(data, event) { Ox.forEach(Ox.makeObject(arguments), function(data, event) {
if ([ if ([
'mousedown', 'mouserepeat', 'anyclick', 'singleclick', 'doubleclick', 'mousedown', 'mouserepeat', 'anyclick', 'singleclick', 'doubleclick',
'dragstart', 'drag', 'dragpause', 'dragend', 'playing' 'dragstart', 'drag', 'dragpause', 'dragend', 'playing', 'progress'
].indexOf(event) == -1) { ].indexOf(event) == -1) {
Ox.print(that.id, self.options.id, 'trigger', event, data); Ox.print(that.id, self.options.id, 'trigger', event, data);
} }

View file

@ -18,9 +18,11 @@ Ox.VideoElement = function(options, self) {
}) })
.options(options || {}) .options(options || {})
.attr({ .attr({
//height: self.options.height,
poster: self.options.poster, poster: self.options.poster,
preload: 'auto', preload: 'auto',
src: self.options.url src: self.options.url,
//width: self.options.width
}) })
.css({ .css({
height: self.options.height + 'px', height: self.options.height + 'px',
@ -28,8 +30,23 @@ Ox.VideoElement = function(options, self) {
}) })
.bind({ .bind({
ended: ended, ended: ended,
canplay: function() {
Ox.print('canplay')
},
durationchange: function() {
Ox.print('durationchange')
},
loadedmetadata: function() { loadedmetadata: function() {
Ox.print('loadedmetadata', self.video.duration)
self.video.currentTime = self.options.position; self.video.currentTime = self.options.position;
that.triggerEvent('loadedmetadata', {
video: self.video
})
},
progress: function() {
that.triggerEvent('progress', {
video: self.video
});
} }
}); });
@ -112,6 +129,7 @@ Ox.VideoElement = function(options, self) {
} }
that.position = function(pos) { that.position = function(pos) {
// fixme: why not use options??
if (arguments.length == 0) { if (arguments.length == 0) {
return self.video.currentTime; return self.video.currentTime;
} else { } else {

View file

@ -0,0 +1,317 @@
Ox.VideoPlayer = function(options, self) {
self = self || {};
var that = Ox.Element({}, self)
.defaults({
height: 192,
position: 0,
timeline: '',
url: '',
width: 256
})
.options(options || {})
.css({
width: self.options.width + 'px',
height: self.options.height + 'px'
//background: 'red'
})
.bind({
mouseenter: showControls,
mouseleave: hideControls
});
self.buffered = [];
self.controlsTimeout;
self.barHeight = 16;
self.outerBarWidth = self.options.width - 96;
self.innerBarWidth = self.outerBarWidth - self.barHeight;
self.markerSize = 12;
self.markerBorderSize = 2;
self.markerOffset = -self.innerBarWidth - self.markerSize / 2;
self.$video = Ox.VideoElement({
height: self.options.height,
paused: true,
url: self.options.url,
width: self.options.width
})
.css({
position: 'absolute',
})
.bindEvent({
loadedmetadata: loadedmetadata,
paused: function(data) {
// called when playback ends
self.$playButton.toggleTitle();
self.$positionMarker.css({
borderColor: 'rgb(192, 192, 192)'
});
},
playing: function(data) {
setPosition(data.position);
},
progress: function(data) {
var buffered = data.video.buffered;
for (var i = 0; i < buffered.length; i++) {
self.buffered[i] = [buffered.start(i), buffered.end(i)];
// fixme: firefox weirdness
if (self.buffered[i][0] > self.buffered[i][1]) {
self.buffered[i][0] = 0;
}
Ox.print(i, self.buffered[i][0], self.buffered[i][1])
}
self.$buffered.attr({
src: getBufferedImageURL()
})
//self.$bar.html(Ox.formatDuration(data.video.buffered.end(data.video.buffered.length - 1)))
}
})
.appendTo(that);
self.$controls = Ox.Bar({
size: self.barHeight
})
.css({
position: 'absolute',
width: self.options.width + 'px',
marginTop: self.options.height - self.barHeight + 'px',
})
.appendTo(that);
self.$buttons = Ox.Element()
.css({
float: 'left',
width: '48px'
})
.appendTo(self.$controls);
self.$playButton = Ox.Button({
style: 'symbol',
title: [
{id: 'play', title: 'play'},
{id: 'pause', title: 'pause'}
],
tooltip: ['Play', 'Pause'],
type: 'image'
})
.css({
borderRadius: 0
})
.bindEvent('click', togglePlay)
.appendTo(self.$buttons);
self.$muteButton = Ox.Button({
style: 'symbol',
title: [
{id: 'mute', title: 'mute'},
{id: 'unmute', title: 'unmute'}
],
tooltip: ['Mute', 'Unmute'],
type: 'image'
})
.css({
borderRadius: 0
})
.bindEvent('click', toggleMute)
.appendTo(self.$buttons);
self.$menuButton = Ox.Button({
id: 'select',
style: 'symbol',
title: 'select',
tooltip: ['Menu'],
type: 'image'
})
.css({
borderRadius: 0
})
.bindEvent('click', togglePlay)
.appendTo(self.$buttons);
self.$outerBar = Ox.Element()
.css({
float: 'left',
width: self.outerBarWidth + 'px',
height: self.barHeight + 'px',
background: 'rgb(0, 0, 0)',
borderRadius: self.barHeight / 2 + 'px'
})
.appendTo(self.$controls);
self.$innerBar = Ox.Element()
.css({
float: 'left',
width: self.innerBarWidth + 'px',
height: self.barHeight + 'px',
marginLeft: self.barHeight / 2 + 'px'
})
.appendTo(self.$outerBar);
self.$timeline = $('<img>')
.attr({
src: self.options.timeline
})
.css({
float: 'left',
width: self.innerBarWidth + 'px',
height: self.barHeight + 'px'
})
.appendTo(self.$innerBar.$element);
self.$buffered = $('<img>')
.attr({
src: getBufferedImageURL()
})
.css({
float: 'left',
marginLeft: -self.innerBarWidth + 'px',
width: self.innerBarWidth + 'px',
height: self.barHeight + 'px',
})
.appendTo(self.$innerBar.$element);
self.$positionMarker = Ox.Element()
.css({
float: 'left',
width: self.markerSize - self.markerBorderSize * 2 + 'px',
height: self.markerSize - self.markerBorderSize * 2 + 'px',
marginTop: (self.barHeight - self.markerSize) / 2 + 'px',
marginLeft: self.markerOffset + 'px',
border: self.markerBorderSize + 'px solid rgb(192, 192, 192)',
borderRadius: self.markerSize - self.markerBorderSize * 2 + 'px',
//background: 'rgba(0, 0, 0, 0.5)',
boxShadow: '0 0 ' + (self.barHeight - self.markerSize) / 2 + 'px black'
})
.appendTo(self.$outerBar);
self.$tooltip = Ox.Tooltip({
animate: false
});
self.$position = Ox.Element()
.css({
float: 'left',
width: '44px',
height: '12px',
padding: '2px',
fontSize: '9px',
textAlign: 'center'
})
.html(Ox.formatDuration(self.options.position))
.appendTo(self.$controls)
function mousedownBar(e) {
setPosition(getPosition(e));
self.$video.position(self.options.position);
}
function mouseleaveBar(e) {
self.$tooltip.hide();
}
function mousemoveBar(e) {
self.$tooltip.options({
title: Ox.formatDuration(getPosition(e))
}).show(e.clientX, e.clientY);
}
function getPosition(e) {
// fixme: no offsetX in firefox???
if ($.browser.mozilla) {
//Ox.print(e, e.layerX - 56)
return Ox.limit(
(e.layerX - 48 - self.barHeight / 2) / self.innerBarWidth * self.duration,
0, self.duration
);
} else {
return Ox.limit(
(e.offsetX - self.barHeight / 2) / self.innerBarWidth * self.duration,
0, self.duration
);
}
}
function setPosition(position) {
self.options.position = position;
//Ox.print(-self.barWidth - 4 + self.barWidth * position / self.duration)
self.$positionMarker.css({
marginLeft: self.innerBarWidth * position / self.duration + self.markerOffset + 'px',
});
self.$position.html(Ox.formatDuration(position));
}
function loadedmetadata(data) {
//self.$position.html(Ox.formatDuration(data.video.duration))
Ox.print('!!!!', data.video.width, data.video.height, data.video.videoWidth, data.video.videoHeight)
self.duration = data.video.duration;
Ox.print('DURATION', Ox.formatDuration(self.duration));
that.gainFocus().bindEvent({
key_space: function() {
self.$playButton.toggleTitle();
togglePlay();
}
});
self.$outerBar.bind({
mousedown: mousedownBar,
mousemove: mousemoveBar,
mouseleave: mouseleaveBar
});
}
function getBufferedImageURL() {
var width = self.innerBarWidth,
height = self.barHeight,
$canvas = $('<canvas>')
.attr({
width: width,
height: height,
}),
canvas = $canvas[0],
context = canvas.getContext('2d');
//Ox.print(width, height)
context.fillStyle = 'rgba(255, 0, 0, 0.5)';
context.fillRect(0, 0, width, height);
var imageData = context.getImageData(0, 0, width, height),
data = imageData.data;
self.buffered.forEach(function(range) {
var left = Math.round(range[0] * width / self.duration),
right = Math.round(range[1] * width / self.duration);
Ox.loop(left, right, function(x) {
Ox.loop(height, function(y) {
index = x * 4 + y * 4 * width;
data[index + 3] = 0;
});
});
});
context.putImageData(imageData, 0, 0);
return canvas.toDataURL();
}
function hideControls() {
self.controlsTimeout = setTimeout(function() {
self.$controls.animate({
opacity: 0
}, 250);
}, 1000);
}
function showControls() {
clearTimeout(self.controlsTimeout);
self.$controls.animate({
opacity: 1
}, 250);
}
function toggleMute() {
self.$video.toggleMute();
}
function togglePlay() {
self.$video.togglePlay();
self.$positionMarker.css({
borderColor: self.$video.paused() ? 'rgb(192, 192, 192)' : 'rgb(255, 255, 255)'
});
}
return that;
};