forked from 0x2620/pandora
302 lines
15 KiB
JavaScript
Executable file
302 lines
15 KiB
JavaScript
Executable file
// vim: et:ts=4:sw=4:sts=4:ft=javascript
|
|
'use strict';
|
|
|
|
Ox.load('UI', {
|
|
debug: false,
|
|
hideScreen: false,
|
|
loadImages: true,
|
|
showScreen: true,
|
|
theme: 'oxdark'
|
|
}, function() {
|
|
var videoKeys = ['duration', 'layers', 'parts', 'posterFrame', 'rightslevel', 'size', 'title', 'videoRatio'];
|
|
window.pandora = new Ox.App({url: '/api/'}).bindEvent({
|
|
load: function(data) {
|
|
Ox.extend(pandora, {
|
|
site: data.site,
|
|
user: data.user.level == 'guest' ? Ox.clone(data.site.user) : data.user,
|
|
$ui: {},
|
|
ui: {
|
|
player: function(options) {
|
|
var that = Ox.Element();
|
|
pandora.user.ui.item = options.item;
|
|
pandora.api.get({id: options.item, keys: videoKeys}, function(result) {
|
|
var data = getVideoOptions(result.data);
|
|
that.append(pandora.$player = Ox.VideoPlayer(Ox.extend({
|
|
censored: data.censored,
|
|
censoredIcon: pandora.site.cantPlay.icon,
|
|
censoredTooltip: pandora.site.cantPlay.text,
|
|
controlsBottom: ['play', 'volume', 'scale'].concat(
|
|
Ox.Fullscreen.available ? ['fullscreen'] : []
|
|
).concat(
|
|
['timeline', 'position', 'settings']
|
|
),
|
|
controlsTooltips: {
|
|
close: 'Close',
|
|
open: 'Watch on ' + pandora.site.site.name
|
|
},
|
|
controlsTop: (options.showCloseButton ? ['close'] : []).concat(
|
|
['title', 'open']
|
|
),
|
|
duration: data.duration,
|
|
enableFind: false,
|
|
enableFullscreen: Ox.Fullscreen.available,
|
|
enableKeyboard: true,
|
|
enableMouse: true,
|
|
enablePosition: true,
|
|
enableSubtitles: true,
|
|
enableTimeline: true,
|
|
enableVolume: true,
|
|
externalControls: false,
|
|
height: window.innerHeight,
|
|
invertHighlight: options.invertHighlight,
|
|
paused: options.paused,
|
|
playInToOut: options.playInToOut,
|
|
position: options.position || 0,
|
|
poster: '/' + options.item + '/' + '96p' + (
|
|
options.position !== void 0 ? options.position
|
|
: options['in'] !== void 0 ? options['in']
|
|
: data.posterFrame
|
|
) +'.jpg',
|
|
resolution: pandora.user.ui.videoResolution,
|
|
showIcon: true,
|
|
showMarkers: false,
|
|
showMilliseconds: 0,
|
|
subtitles: data.subtitles,
|
|
timeline: options.playInToOut ? function(size, i) {
|
|
return '/' + options.item
|
|
+ '/timelineantialias'
|
|
+ size + 'p' + i + '.jpg'
|
|
} : '/' + options.item + '/' + 'timeline16p.png',
|
|
title: result.data.title,
|
|
video: data.video,
|
|
width: window.innerWidth
|
|
}, options['in'] ? {
|
|
'in': options['in'],
|
|
} : {}, options.out ? {
|
|
out: options.out
|
|
} : {}))
|
|
.bindEvent(Ox.extend({
|
|
open: function() {
|
|
pandora.$player.options({paused: true});
|
|
var url = document.location.protocol + '//'
|
|
+ document.location.hostname + '/'
|
|
+ options.item + '/'
|
|
+ Ox.formatDuration(pandora.$player.options('position'));
|
|
window.open(url, '_blank');
|
|
},
|
|
resolution: function(data) {
|
|
pandora.api.setUI({'videoResolution': data.resolution});
|
|
},
|
|
fullscreen: function(data) {
|
|
Ox.Fullscreen.toggle();
|
|
}
|
|
}, ((options['in'] || options.out) && !options.playInToOut) ? {
|
|
playing: checkRange,
|
|
position: checkRange
|
|
} : {}))
|
|
.bindEvent(function(data, event) {
|
|
if (window.parent) {
|
|
window.parent.postMessage(JSON.stringify({
|
|
event: event,
|
|
id: options.id
|
|
}), '*');
|
|
}
|
|
})
|
|
);
|
|
Ox.UI.hideScreen();
|
|
});
|
|
return that;
|
|
},
|
|
timeline: function(options) {
|
|
var that = Ox.Element();
|
|
pandora.user.ui.item = options.item;
|
|
pandora.api.get({id: options.item, keys: videoKeys}, function(result) {
|
|
Ox.UI.hideScreen();
|
|
var data = getVideoOptions(result.data),
|
|
ui = pandora.user.ui;
|
|
that.append(pandora.player = Ox.VideoTimelinePlayer({
|
|
censored: data.censored,
|
|
censoredIcon: pandora.site.cantPlay.icon,
|
|
censoredTooltip: pandora.site.cantPlay.text,
|
|
cuts: data.cuts || [],
|
|
duration: data.duration,
|
|
followPlayer: ui.followPlayer,
|
|
getFrameURL: function(position) {
|
|
return '/' + ui.item + '/' + ui.videoResolution + 'p' + position + '.jpg';
|
|
},
|
|
getLargeTimelineURL: function(type, i) {
|
|
return '/' + ui.item + '/timeline' + type + '64p' + i + '.jpg';
|
|
},
|
|
height: that.height(),
|
|
muted: ui.videoMuted,
|
|
'in': options['in'],
|
|
out: options.out,
|
|
paused: options.paused,
|
|
position: options['in'],
|
|
resolution: Ox.min(pandora.site.video.resolutions),
|
|
smallTimelineURL: '/' + ui.item + '/timeline16p.jpg',
|
|
subtitles: data.subtitles,
|
|
timeline: ui.videoTimeline,
|
|
timelines: pandora.site.timelines,
|
|
video: data.video,
|
|
videoRatio: data.videoRatio,
|
|
volume: ui.videoVolume,
|
|
width: that.width()
|
|
})
|
|
.bindEvent({
|
|
playing: checkRange,
|
|
position: checkRange,
|
|
resolution: function(data) {
|
|
pandora.api.setUI({'videoResolution': data.resolution});
|
|
},
|
|
})
|
|
);
|
|
});
|
|
return that;
|
|
}
|
|
}
|
|
});
|
|
function checkRange(data) {
|
|
if (
|
|
data.position < options['in'] - 0.04
|
|
|| data.position > options.out
|
|
) {
|
|
if (!pandora.$player.options('paused')) {
|
|
pandora.$player.togglePaused();
|
|
if (data.position > options.out) {
|
|
data.position = options['in'] - 0.05;
|
|
}
|
|
}
|
|
pandora.$player.options({
|
|
position: data.position < options['in'] - 0.04
|
|
? options['in'] : options.out
|
|
});
|
|
}
|
|
}
|
|
Ox.extend(pandora.user, {
|
|
videoFormat: Ox.getVideoFormat(pandora.site.video.formats)
|
|
});
|
|
var options = parseQuery();
|
|
if (['video', 'player'].indexOf(options.view) > -1) {
|
|
pandora.$ui.player = pandora.ui.player(options)
|
|
.css({top: 0, bottom: 0, left: 0, right: 0, position: 'absolute'})
|
|
.appendTo(document.body);
|
|
} else if (options.view == 'timeline') {
|
|
pandora.$ui.timeline = pandora.ui.timeline(options)
|
|
.css({top: 0, bottom: 0, left: 0, right: 0, position: 'absolute'})
|
|
.appendTo(document.body);
|
|
}
|
|
}
|
|
});
|
|
|
|
function getVideoOptions(data) {
|
|
var canPlayClips = data.editable || pandora.hasCapability('canPlayClips') >= data.rightslevel,
|
|
canPlayVideo = data.editable || pandora.hasCapability('canPlayVideo') >= data.rightslevel,
|
|
options = {},
|
|
subtitlesLayer = pandora.site.layers.filter(function(layer) {
|
|
return layer.isSubtitles;
|
|
})[0];
|
|
options.censored = canPlayVideo ? []
|
|
: canPlayClips ? (
|
|
options.subtitles.length
|
|
? options.subtitles.map(function(subtitle, i) {
|
|
return {
|
|
'in': i == 0 ? 0 : options.subtitles[i - 1].out,
|
|
out: subtitle['in']
|
|
};
|
|
}).concat(
|
|
[{'in': Ox.last(options.subtitles).out, out: data.duration}]
|
|
).filter(function(censored) {
|
|
// don't include gaps shorter than one second
|
|
return censored.out - censored['in'] >= 1;
|
|
})
|
|
: Ox.range(0, data.duration - 5, 60).map(function(position) {
|
|
return {
|
|
'in': position + 5,
|
|
out: Math.min(position + 60, data.duration)
|
|
};
|
|
})
|
|
)
|
|
: [{'in': 0, out: data.duration}];
|
|
options.duration = data.duration;
|
|
options.layers = [];
|
|
pandora.site.layers.forEach(function(layer, i) {
|
|
options.layers[i] = Ox.extend({}, layer, {
|
|
items: data.layers[layer.id].map(function(annotation) {
|
|
annotation.duration = Math.abs(annotation.out - annotation['in']);
|
|
annotation.editable = annotation.editable
|
|
|| annotation.user == pandora.user.username
|
|
|| pandora.hasCapability('canEditAnnotations');
|
|
return annotation;
|
|
})
|
|
});
|
|
});
|
|
options.posterFrame = data.posterFrame;
|
|
options.subtitles = subtitlesLayer ? data.layers[subtitlesLayer.id].map(function(subtitle) {
|
|
return {
|
|
id: subtitle.id,
|
|
'in': subtitle['in'],
|
|
out: subtitle.out,
|
|
text: subtitle.value.replace(/\n/g, ' ').replace(/<br\/?>/g, '\n')
|
|
};
|
|
}) : [];
|
|
options.video = {};
|
|
pandora.site.video.resolutions.forEach(function(resolution) {
|
|
options.video[resolution] = Ox.range(data.parts).map(function(i) {
|
|
return getVideoURL(data.item || pandora.user.ui.item, resolution, i + 1);
|
|
});
|
|
});
|
|
options.videoRatio = data.videoRatio;
|
|
return options;
|
|
}
|
|
|
|
function getVideoURL(id, resolution, part) {
|
|
var uid = Ox.uid(),
|
|
prefix = pandora.site.site.videoprefix
|
|
.replace('{id}', id)
|
|
.replace('{part}', part)
|
|
.replace('{resolution}', resolution)
|
|
.replace('{uid}', uid)
|
|
.replace('{uid42}', uid % 42);
|
|
return prefix + '/' + id + '/' + resolution + 'p' + part + '.' + pandora.user.videoFormat;
|
|
}
|
|
|
|
function parseQuery() {
|
|
var vars = window.location.search.length
|
|
? window.location.search.slice(1).split('&')
|
|
: [],
|
|
query = {
|
|
item: window.location.pathname.slice(1).split('/')[0]
|
|
},
|
|
defaults = {
|
|
invertHighlight: true,
|
|
paused: true,
|
|
playInToOut: true,
|
|
view: 'video',
|
|
},
|
|
options;
|
|
vars.forEach(function(v) {
|
|
var kv = v.split('='), k = kv[0], v = kv[1];
|
|
query[k] = Ox.decodeURIComponent(v);
|
|
if (query[k] == 'true') {
|
|
query[k] = true;
|
|
} else if (query[k] == 'false') {
|
|
query[k] = false;
|
|
} else if (query[k].match(/^[\d\.]+$/)) {
|
|
query[k] = parseFloat(query[k]);
|
|
}
|
|
if (['in', 'out'].indexOf(k) > -1 && v.indexOf(':') > -1) {
|
|
query[k] = Ox.parseDuration(query[k]);
|
|
}
|
|
});
|
|
options = Ox.extend({}, defaults, query);
|
|
if (!options.position) {
|
|
options.position = options['in'] || 0;
|
|
}
|
|
if (!options['in'] && !options.out) {
|
|
options.playInToOut = false;
|
|
}
|
|
return options;
|
|
}
|
|
|
|
});
|