embyLaunchIINA

emby launch extetnal player

// ==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]
  }
})()