// ==UserScript==
// @name embyLaunchIINA
// @name:en embyLaunchIINA
// @name:zh embyLaunchIINA
// @name:zh-CN embyLaunchIINA
// @namespace http://tampermonkey.net/
// @version 0.0.11
// @description emby launch extetnal player
// @description:zh-cn emby调用外部播放器
// @description:en emby to external player
// @license MIT
// @match *://*/web/index.html*
// ==/UserScript==
// 修改自 https://github.com/bpking1/embyExternalUrl 自用
;(function () {
const os = getOS()
let timeid = null
function init() {
if(timeid){
clearTimeout(timeid)
timeid = null
}
let playBtns = document.getElementById('externalPlayer')
if (playBtns) {
playBtns.remove()
}
let palyButton = document.querySelector(
"div[is='emby-scroller']:not(.hide) .mainDetailButtons .btnPlay"
)
if (!palyButton) {
return
}
let externalPlayer = '' // IINA PotPlayer MXPlayer NPlayer VLC Infuse MPV
switch (os) {
case 'macOS':
externalPlayer = 'IINA'
break
case 'windows':
externalPlayer = 'PotPlayer'
break
case 'android':
case 'ios':
externalPlayer = 'NPlayer'
break
default:
externalPlayer = 'MPV'
break
}
let buttonhtml = `
<button id="externalPlayer" type="button" class="raised detailButton emby-button detailButton-primary" title="externalPlayer"> <i class="md-icon detailButton-icon button-icon button-icon-left icon-externalPlayer"> </i> <span class="button-text">${externalPlayer}</span> </button>
`
let buttonStyle = `
background: url(${getIconsExt(
externalPlayer
)})no-repeat;background-size: 100% 100%;font-size: 1.4em;
`
palyButton.insertAdjacentHTML('afterend', buttonhtml)
document.querySelector(
"div[is='emby-scroller']:not(.hide) .icon-externalPlayer"
).style.cssText += buttonStyle
document.querySelector("div[is='emby-scroller']:not(.hide) #externalPlayer").onclick = () => {
// 调用外部播放器
eval('emby'+externalPlayer+'()')
// 30秒后设置为已播放
setPlayed()
}
}
function showFlag() {
// itemMiscInfo-primary
// 评分,上映日期信息栏
let mediaInfoPrimary = document.querySelector(
"div[is='emby-scroller']:not(.hide) .mediaInfoPrimary:not(.hide)"
)
// 创建录制按钮
let btnManualRecording = document.querySelector(
"div[is='emby-scroller']:not(.hide) .btnManualRecording:not(.hide)"
)
return !!mediaInfoPrimary || !!btnManualRecording
}
async function getItemInfo() {
let userId = ApiClient._serverInfo.UserId;
let itemId = /\?id=([A-Za-z0-9]+)/.exec(window.location.hash)[1];
let response = await ApiClient.getItem(userId, itemId);
// 继续播放当前剧集的下一集
if (response.Type == "Series") {
let seriesNextUpItems = await ApiClient.getNextUpEpisodes({ SeriesId: itemId, UserId: userId });
if (seriesNextUpItems.Items.length > 0) {
console.log("nextUpItemId: " + seriesNextUpItems.Items[0].Id);
return await ApiClient.getItem(userId, seriesNextUpItems.Items[0].Id);
}
}
// 播放当前季season的第一集
if (response.Type == "Season") {
let seasonItems = await ApiClient.getItems(userId, { parentId: itemId });
console.log("seasonItemId: " + seasonItems.Items[0].Id);
return await ApiClient.getItem(userId, seasonItems.Items[0].Id);
}
// 播放当前集或电影
if (response.MediaSources?.length > 0) {
console.log("itemId: " + itemId);
return response;
}
// 默认播放第一个,集/播放列表第一个媒体
let firstItems = await ApiClient.getItems(userId, { parentId: itemId, Recursive: true, IsFolder: false, Limit: 1 });
console.log("firstItemId: " + firstItems.Items[0].Id);
return await ApiClient.getItem(userId, firstItems.Items[0].Id);
}
function getSeek(position) {
let ticks = position * 10000
let parts = [],
hours = ticks / 36e9
;(hours = Math.floor(hours)) && parts.push(hours)
let minutes = (ticks -= 36e9 * hours) / 6e8
;(ticks -= 6e8 * (minutes = Math.floor(minutes))),
minutes < 10 && hours && (minutes = '0' + minutes),
parts.push(minutes)
let seconds = ticks / 1e7
return (
(seconds = Math.floor(seconds)) < 10 && (seconds = '0' + seconds),
parts.push(seconds),
parts.join(':')
)
}
function getSubPath(mediaSource) {
let selectSubtitles = document.querySelector(
"div[is='emby-scroller']:not(.hide) select.selectSubtitles"
)
let subTitlePath = ''
//返回选中的外挂字幕
if (selectSubtitles && selectSubtitles.value > 0) {
let SubIndex = mediaSource.MediaStreams.findIndex(
m => m.Index == selectSubtitles.value && m.IsExternal
)
if (SubIndex > -1) {
let subtitleCodec = mediaSource.MediaStreams[SubIndex].Codec
subTitlePath = `/${mediaSource.Id}/Subtitles/${selectSubtitles.value}/Stream.${subtitleCodec}`
}
} else {
//默认尝试返回第一个外挂中文字幕
let chiSubIndex = mediaSource.MediaStreams.findIndex(m => m.Language == 'chi' && m.IsExternal)
if (chiSubIndex > -1) {
let subtitleCodec = mediaSource.MediaStreams[chiSubIndex].Codec
subTitlePath = `/${mediaSource.Id}/Subtitles/${chiSubIndex}/Stream.${subtitleCodec}`
} else {
//尝试返回第一个外挂字幕
let externalSubIndex = mediaSource.MediaStreams.findIndex(m => m.IsExternal)
if (externalSubIndex > -1) {
let subtitleCodec = mediaSource.MediaStreams[externalSubIndex].Codec
subTitlePath = `/${mediaSource.Id}/Subtitles/${externalSubIndex}/Stream.${subtitleCodec}`
}
}
}
return subTitlePath
}
async function getEmbyMediaInfo() {
let itemInfo = await getItemInfo()
let mediaSourceId = itemInfo.MediaSources[0].Id
let selectSource = document.querySelector(
"div[is='emby-scroller']:not(.hide) select.selectSource:not([disabled])"
)
if (selectSource && selectSource.value.length > 0) {
mediaSourceId = selectSource.value
}
//let selectAudio = document.querySelector("div[is='emby-scroller']:not(.hide) select.selectAudio:not([disabled])");
let mediaSource = itemInfo.MediaSources.find(m => m.Id == mediaSourceId)
let domain = `${ApiClient._serverAddress}/emby/videos/${itemInfo.Id}`
let subPath = getSubPath(mediaSource)
let subUrl = subPath.length > 0 ? `${domain}${subPath}?api_key=${ApiClient.accessToken()}` : ''
let streamUrl = `${domain}/`
if (mediaSource.IsInfiniteStream) {
streamUrl += `master.m3u8`
} else {
streamUrl += `stream.${mediaSource.Container}`
}
streamUrl += `?api_key=${ApiClient.accessToken()}&Static=true&MediaSourceId=${mediaSourceId}&DeviceId=${ApiClient._deviceId}`
let position = parseInt(itemInfo.UserData.PlaybackPositionTicks / 10000)
let intent = await getIntent(mediaSource, position)
console.log(streamUrl, subUrl, intent)
return {
streamUrl: streamUrl,
subUrl: subUrl,
intent: intent,
}
}
async function getIntent(mediaSource, position) {
// 直播节目查询items接口没有path
let title = mediaSource.IsInfiniteStream ? mediaSource.Name : mediaSource.Path.split('/').pop()
let externalSubs = mediaSource.MediaStreams.filter(m => m.IsExternal == true)
let subs = '' //要求是android.net.uri[] ?
let subs_name = ''
let subs_filename = ''
let subs_enable = ''
if (externalSubs) {
subs_name = externalSubs.map(s => s.DisplayTitle)
subs_filename = externalSubs.map(s => s.Path.split('/').pop())
}
return {
title: title,
position: position,
subs: subs,
subs_name: subs_name,
subs_filename: subs_filename,
subs_enable: subs_enable,
}
}
// 打勾标记已观看
function setPlayed() {
const btnPlaystateElement = document.querySelector(
"div[is='emby-scroller']:not(.hide) .mainDetailButtons .btnPlaystate"
)
if (!btnPlaystateElement) {
return
}
const playstate = btnPlaystateElement.getAttribute('data-played')
if (playstate === 'false') {
timeid = setTimeout(() => {
btnPlaystateElement.click()
}, 60000)
}
}
async function embyPotPlayer() {
let mediaInfo = await getEmbyMediaInfo()
let intent = mediaInfo.intent
let poturl = `potplayer://${encodeURI(mediaInfo.streamUrl)} /sub=${encodeURI(mediaInfo.subUrl)} /current /seek=${getSeek(intent.position)} /title=${intent.title}`;
console.log(poturl)
window.open(poturl, '_blank')
}
//https://wiki.videolan.org/Android_Player_Intents/
async function embyVLC() {
let mediaInfo = await getEmbyMediaInfo()
let intent = mediaInfo.intent
//android subtitles: https://code.videolan.org/videolan/vlc-android/-/issues/1903
let vlcUrl = `intent:${encodeURI(
mediaInfo.streamUrl
)}#Intent;package=org.videolan.vlc;type=video/*;S.subtitles_location=${encodeURI(
mediaInfo.subUrl
)};S.title=${encodeURI(intent.title)};i.position=${intent.position};end`
if (os == 'windows') {
//桌面端需要额外设置,参考这个项目: https://github.com/stefansundin/vlc-protocol
vlcUrl = `vlc://${encodeURI(mediaInfo.streamUrl)}`
}
if (os == 'ios') {
//https://code.videolan.org/videolan/vlc-ios/-/commit/55e27ed69e2fce7d87c47c9342f8889fda356aa9
vlcUrl = `vlc-x-callback://x-callback-url/stream?url=${encodeURIComponent(
mediaInfo.streamUrl
)}&sub=${encodeURIComponent(mediaInfo.subUrl)}`
}
console.log(vlcUrl)
window.open(vlcUrl, '_blank')
}
//https://github.com/iina/iina/issues/1991
async function embyIINA() {
let mediaInfo = await getEmbyMediaInfo()
let iinaUrl = `iina://weblink?url=${encodeURIComponent(mediaInfo.streamUrl)}&new_window=1`
console.log(`iinaUrl= ${iinaUrl}`)
window.open(iinaUrl, '_blank')
}
//https://sites.google.com/site/mxvpen/api
async function embyMXPlayer() {
let mediaInfo = await getEmbyMediaInfo()
let intent = mediaInfo.intent
//mxPlayer free
let mxUrl = `intent:${encodeURI(
mediaInfo.streamUrl
)}#Intent;package=com.mxtech.videoplayer.ad;S.title=${encodeURI(intent.title)};i.position=${
intent.position
};end`
//mxPlayer Pro
//let mxUrl = `intent:${encodeURI(mediaInfo.streamUrl)}#Intent;package=com.mxtech.videoplayer.pro;S.title=${encodeURI(intent.title)};i.position=${intent.position};end`;
console.log(mxUrl)
window.open(mxUrl, '_blank')
}
async function embyNPlayer() {
let mediaInfo = await getEmbyMediaInfo()
let nUrl =
os == 'macOS'
? `nplayer-mac://weblink?url=${encodeURIComponent(mediaInfo.streamUrl)}&new_window=1`
: `nplayer-${encodeURI(mediaInfo.streamUrl)}`
console.log(nUrl)
window.open(nUrl, '_blank')
}
//infuse
async function embyInfuse() {
let mediaInfo = await getEmbyMediaInfo()
let infuseUrl = `infuse://x-callback-url/play?url=${encodeURIComponent(mediaInfo.streamUrl)}`
console.log(`infuseUrl= ${infuseUrl}`)
window.open(infuseUrl, '_blank')
}
//MPV
async function embyMPV() {
let mediaInfo = await getEmbyMediaInfo()
//桌面端需要额外设置,使用这个项目: https://github.com/akiirui/mpv-handler
let streamUrl64 = btoa(mediaInfo.streamUrl)
.replace(/\//g, '_')
.replace(/\+/g, '-')
.replace(/\=/g, '')
let MPVUrl = `mpv://play/${streamUrl64}`
if (mediaInfo.subUrl.length > 0) {
let subUrl64 = btoa(mediaInfo.subUrl)
.replace(/\//g, '_')
.replace(/\+/g, '-')
.replace(/\=/g, '')
MPVUrl = `mpv://play/${streamUrl64}/?subfile=${subUrl64}`
}
if (os == 'ios' || os == 'android' || os == 'macOS') {
MPVUrl = `mpv://${encodeURI(mediaInfo.streamUrl)}`
}
console.log(MPVUrl)
window.open(MPVUrl, '_blank')
}
function getOS() {
let u = navigator.userAgent
if (!!u.match(/compatible/i) || u.match(/Windows/i)) {
return 'windows'
} else if (!!u.match(/Macintosh/i) || u.match(/MacIntel/i)) {
return 'macOS'
} else if (!!u.match(/iphone/i) || u.match(/Ipad/i)) {
return 'ios'
} else if (u.match(/android/i)) {
return 'android'
} else if (u.match(/Ubuntu/i)) {
return 'Ubuntu'
} else {
return 'other'
}
}
// monitor dom changements
document.addEventListener('viewbeforeshow', function (e) {
console.log('viewbeforeshow', e)
if (e.detail.contextPath.startsWith('/item?id=')) {
const mutation = new MutationObserver(function () {
if (showFlag()) {
init()
mutation.disconnect()
}
})
mutation.observe(document.body, {
childList: true,
characterData: true,
subtree: true,
})
}
})
function getIconsExt(type) {
const iconsExt = {
IINA: `

`,
PotPlayer : `

`,
VLC: `

`,
IINA: `

`,
NPlayer: `

`,
MXPlayer: `

`,
Infuse: `

`,
MPV: `

`,
}
return iconsExt[type]
}
})()