embyLaunchPotplayer

try to take over the world!

目前為 2022-05-24 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         embyLaunchPotplayer
// @name:en      embyLaunchPotplayer
// @name:zh      embyLaunchPotplayer
// @name:zh-CN   embyLaunchPotplayer
// @namespace    http://tampermonkey.net/
// @version      1.0.2
// @description  try to take over the world!
// @description:zh-cn emby调用外部播放器
// @license      MIT
// @author       @bpking
// @include      */web/index.html
// ==/UserScript==


let api_key = '';

const reg = /\/[a-z]{2,}\/\S*?id=/;


let timer = setInterval(function () {
  let potplayer = document.querySelectorAll("div[is='emby-scroller']:not(.hide) #embyPot")[0];
  if (!potplayer) {
    let mainDetailButtons = document.querySelectorAll("div[is='emby-scroller']:not(.hide) .mainDetailButtons")[0];
    if (mainDetailButtons) {
      let buttonhtml = `<div class ="detailButtons  flex align-items-flex-start flex-wrap-wrap">
                  <button id="embyPot" type="button" class="detailButton  emby-button emby-button-backdropfilter raised-backdropfilter detailButton-primary" title="Potplayer"> <div class="detailButton-content"> <i class="md-icon detailButton-icon button-icon button-icon-left"></i>  <span class="button-text">Pot</span> </div> </button>
                  <button id="embyVlc" type="button" class="detailButton  emby-button emby-button-backdropfilter raised-backdropfilter detailButton-primary" title="VLC"> <div class="detailButton-content"> <i class="md-icon detailButton-icon button-icon button-icon-left"></i>  <span class="button-text">VLC</span>  </div> </button>
                  <button id="embyIINA" type="button" class="detailButton  emby-button emby-button-backdropfilter raised-backdropfilter detailButton-primary" title="IINA"> <div class="detailButton-content"> <i class="md-icon detailButton-icon button-icon button-icon-left"></i>  <span class="button-text">IINA</span> </div> </button>
                  <button id="embyNPlayer" type="button" class="detailButton  emby-button emby-button-backdropfilter raised-backdropfilter detailButton-primary" title="NPlayer"> <div class="detailButton-content"> <i class="md-icon detailButton-icon button-icon button-icon-left"></i>  <span class="button-text">NPlayer</span> </div> </button>
                  <button id="embyMX" type="button" class="detailButton  emby-button emby-button-backdropfilter raised-backdropfilter detailButton-primary" title="MXPlayer"> <div class="detailButton-content"> <i class="md-icon detailButton-icon button-icon button-icon-left"></i>  <span class="button-text">MX</span> </div> </button>
                  <button id="embyCopyUrl" type="button" class="detailButton  emby-button emby-button-backdropfilter raised-backdropfilter detailButton-primary" title="复制链接"> <div class="detailButton-content"> <i class="md-icon detailButton-icon button-icon button-icon-left"></i>  <span class="button-text">复制链接</span> </div> </button>
                      </div>`
      mainDetailButtons.insertAdjacentHTML('afterend', buttonhtml)
      document.querySelector("div[is='emby-scroller']:not(.hide) #embyPot").onclick = embyPot;
      //document.querySelector("div[is='emby-scroller']:not(.hide) #embyPotAdd").onclick = embyPotAdd;
      document.querySelector("div[is='emby-scroller']:not(.hide) #embyIINA").onclick = embyIINA;
      document.querySelector("div[is='emby-scroller']:not(.hide) #embyNPlayer").onclick = embyNPlayer;
      document.querySelector("div[is='emby-scroller']:not(.hide) #embyMX").onclick = embyMX;
      document.querySelector("div[is='emby-scroller']:not(.hide) #embyCopyUrl").onclick = embyCopyUrl;
      document.querySelector("div[is='emby-scroller']:not(.hide) #embyVlc").onclick = embyVlc;
      api_key = ApiClient.accessToken();
    }
  }
}, 1000)

async function getItemInfo() {
  let itemInfoUrl = window.location.href.replace(reg, "/emby/Items/").split('&')[0] + "/PlaybackInfo?api_key=" + api_key;
  console.log("itemInfo: " + itemInfoUrl);
  let response = await fetch(itemInfoUrl);
  if (response.ok) {
    return await response.json();
  } else {
    alert("获取视频信息失败,检查emby网络  " + response.status + " " + response.statusText);
    throw new Error(response.statusText);
  }
}


async function getPositon() {
  let userDataUrl = window.location.href.replace(reg, `/emby/Users/${ApiClient._serverInfo.UserId}/Items/`).split('&')[0] + "?api_key=" + api_key;
  console.log("userData: " + userDataUrl);
  let response = await fetch(userDataUrl);
  if (response.ok) {
    let data = await response.json();
    return parseInt(data.UserData.PlaybackPositionTicks / 10000);
  } else {
    return 0;
  }
}

async function getSeek() {
  var ticks = await getPositon() * 10000;
  var parts = []
    , hours = ticks / 36e9;
  (hours = Math.floor(hours)) && parts.push(hours);
  var minutes = (ticks -= 36e9 * hours) / 6e8;
  ticks -= 6e8 * (minutes = Math.floor(minutes)),
    minutes < 10 && hours && (minutes = "0" + minutes),
    parts.push(minutes);
  var seconds = ticks / 1e7;
  return (seconds = Math.floor(seconds)) < 10 && (seconds = "0" + seconds),
    parts.push(seconds),
    parts.join(":")
}

function getSubUrl(mediaSource) {
  const selectSubtitles = document.querySelector("div[is='emby-scroller']:not(.hide) select.selectSubtitles");
  let subTitleUrl = '';
  if (selectSubtitles && selectSubtitles.value > 0) {
    if (mediaSource.MediaStreams[selectSubtitles.value].IsExternal) {
      let subtitleCodec = mediaSource.MediaStreams[selectSubtitles.value].Codec;
      const domain = window.location.href.replace(reg, "/emby/videos/").split('&')[0];
      subTitleUrl = `${domain}/${mediaSource.Id}/Subtitles/${selectSubtitles.value}/Stream.${subtitleCodec}?api_key=${api_key}`;
      console.log(subTitleUrl);
    }
  }
  return subTitleUrl;
}


async function getEmbyMediaUrl() {
  const mediaSourceId = document.querySelector("div[is='emby-scroller']:not(.hide) select.selectSource").value;
  //let selectAudio = document.querySelector("div[is='emby-scroller']:not(.hide) select.selectAudio");
  const itemInfo = await getItemInfo();
  const mediaSource = itemInfo.MediaSources.find(m => m.Id == mediaSourceId);
  let PlaySessionId = itemInfo.PlaySessionId;
  let subUrl = await getSubUrl(mediaSource);
  //let streamUrl = ApiClient._serverAddress +'/emby'+ mediaSource.DirectStreamUrl;
  const domain = window.location.href.replace(reg, "/emby/videos/").split('&')[0];
  let streamUrl = `${domain}/stream.${mediaSource.Container}?api_key=${api_key}&Static=true&MediaSourceId=${mediaSourceId}&PlaySessionId=${PlaySessionId}`;
  const intent = await getIntent(mediaSource);
  console.log(streamUrl, subUrl, intent);
  return Array(streamUrl, subUrl, intent);
}

async function getIntent(mediaSource) {
  const title = mediaSource.Path.split('/').pop();
  let position = await getPositon();

  let externalSubs = mediaSource.MediaStreams.filter(m => m.IsExternal == true);
  const subs = ''; //要求是android.net.uri[] ?
  let subs_name = '';
  let subs_filename = '';
  const 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
  };
}

async function embyPot() {
  let mediaUrl = await getEmbyMediaUrl();
  let poturl = `potplayer://${encodeURI(mediaUrl[0])} /sub=${encodeURI(mediaUrl[1])} /current /seek=${await getSeek()}`;
  console.log(poturl);
  window.open(poturl, "_blank");
}
//稍后播放,添加至potPlayer播放列表
async function embyPotAdd() {
  let mediaUrl = await getEmbyMediaUrl();
  let poturl = `potplayer://${encodeURI(mediaUrl[0])} /sub=${encodeURI(mediaUrl[1])} /current /add /seek=${await getSeek()}`;
  console.log(poturl);
  window.open(poturl, "_blank");
}
async function embyVlc() {
  let mediaUrl = await getEmbyMediaUrl();
  let intent = mediaUrl[2];
  let vlcUrl = `intent:${encodeURI(mediaUrl[0])}#Intent;package=org.videolan.vlc;type=video/*;S.subtitles_location=${encodeURI(mediaUrl[1])};S.title=${intent.title};i.position=${intent.position};end`;
  if (getOS() == "windows") {
    vlcUrl = `vlc://${encodeURI(mediaUrl[0])}`;
  }
  if (getOS() == 'ios') {
    vlcUrl = `vlc-x-callback://x-callback-url/stream?url=${encodeURI(mediaUrl[0])}&sub=${encodeURI(mediaUrl[1])}`;
  }
  console.log(vlcUrl);
  window.open(vlcUrl, "_blank");
}
async function embyIINA() {
  let mediaUrl = await getEmbyMediaUrl();
  let iinaUrl = `iina://weblink?url=${escape(mediaUrl[0])}&new_window=1`;
  console.log(`iinaUrl= ${iinaUrl}`);
  window.open(iinaUrl, "_blank");
}
async function embyMX() {
  let mediaUrl = await getEmbyMediaUrl();
  const intent = mediaUrl[2];
  //mxPlayer free
  let mxUrl = `intent:${encodeURI(mediaUrl[0])}#Intent;package=com.mxtech.videoplayer.ad;S.title=${intent.title};i.position=${intent.position};end`;
  //mxPlayer Pro
  //let mxUrl = `intent:${encodeURI(mediaUrl[0])}#Intent;package=com.mxtech.videoplayer.pro;S.title=${intent.title};i.position=${intent.position};end`;
  console.log(mxUrl);
  window.open(mxUrl, "_blank");
}
async function embyNPlayer() {
  let mediaUrl = await getEmbyMediaUrl();
  let nUrl = `nplayer-${encodeURI(mediaUrl[0])}`;
  console.log(nUrl);
  window.open(nUrl, "_blank");
}
async function embyCopyUrl() {
  let mediaUrl = await getEmbyMediaUrl();
  let textarea = document.createElement('textarea');
  document.body.appendChild(textarea);
  textarea.style.position = 'absolute';
  textarea.style.clip = 'rect(0 0 0 0)';
  textarea.value = mediaUrl[0];
  textarea.select();
  if (document.execCommand('copy', true)) {
    console.log(`copyUrl = ${mediaUrl[0]}`);
    this.innerText = '复制成功';
  }
  //need https
  // if (navigator.clipboard) {
  //     console.log('test');
  //     navigator.clipboard.writeText(mediaUrl[0]).then(() => {
  //          console.log(`copyUrl = ${mediaUrl[0]}`);
  //          this.innerText = '复制成功';
  //     })
  // }
}
function getOS() {
  const 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'
  }
}