background-audio.js 3.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11
import {
  onMethod,
  invokeMethod
} from '../../platform'

const eventNames = [
  'canplay',
  'play',
  'pause',
  'stop',
  'ended',
12
  'timeUpdate',
13 14 15 16 17 18 19 20 21 22
  'prev',
  'next',
  'error',
  'waiting'
]
const callbacks = {}
eventNames.forEach(name => {
  callbacks[name] = []
})

23
const props = [
24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
  {
    name: 'duration',
    readonly: true
  },
  {
    name: 'currentTime',
    readonly: true
  },
  {
    name: 'paused',
    readonly: true
  },
  {
    name: 'src',
    cache: true
  },
  {
    name: 'startTime',
    default: 0,
    cache: true
  },
  {
    name: 'buffered',
    readonly: true
  },
  {
    name: 'title',
    cache: true
  },
  {
    name: 'epname',
    cache: true
  },
  {
    name: 'singer',
    cache: true
  },
  {
    name: 'coverImgUrl',
    cache: true
  },
  {
    name: 'webUrl',
    cache: true
  },
  {
    name: 'protocol',
    readonly: true,
    default: 'http'
73 74 75 76
  },
  {
    name: 'playbackRate',
    default: 1,
Q
qiang 已提交
77
    cache: true
78 79
  }
]
80 81

const backgroundEvents = ['prev', 'next']
82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98

class BackgroundAudioManager {
  constructor () {
    this._options = {}
    onMethod('onBackgroundAudioStateChange', ({
      state,
      errMsg,
      errCode
    }) => {
      callbacks[state].forEach(callback => {
        if (typeof callback === 'function') {
          callback(state === 'error' ? {
            errMsg,
            errCode
          } : {})
        }
      })
99 100 101 102 103 104 105 106 107
    })
    backgroundEvents.forEach((name) => {
      onMethod(`onBackgroundAudio${name[0].toUpperCase() + name.substr(1)}`, () => {
        callbacks[name].forEach(callback => {
          if (typeof callback === 'function') {
            callback({})
          }
        })
      })
108 109 110 111 112 113 114 115 116 117 118 119 120 121
    })
    props.forEach(item => {
      const name = item.name
      const data = {
        get () {
          const result = item.cache ? this._options : invokeMethod('getBackgroundAudioState')
          return name in result ? result[name] : item.default
        }
      }
      if (!item.readonly) {
        data.set = function (value) {
          this._options[name] = value
          invokeMethod('setBackgroundAudioState', Object.assign({}, this._options, {
            audioId: this.id
122
          }), name)
123 124 125 126
        }
      }
      Object.defineProperty(this, name, data)
    })
127
  }
fxy060608's avatar
fxy060608 已提交
128

129 130
  play () {
    this._operate('play')
131
  }
fxy060608's avatar
fxy060608 已提交
132

133 134
  pause () {
    this._operate('pause')
135
  }
fxy060608's avatar
fxy060608 已提交
136

137 138
  stop () {
    this._operate('stop')
139
  }
fxy060608's avatar
fxy060608 已提交
140

141
  seek (position) {
雪洛's avatar
雪洛 已提交
142
    this._operate('seek', {
143 144
      currentTime: position
    })
145
  }
fxy060608's avatar
fxy060608 已提交
146

147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164
  _operate (type, options) {
    invokeMethod('operateBackgroundAudio', Object.assign({}, options, {
      operationType: type
    }))
  }
}

eventNames.forEach(item => {
  const name = item[0].toUpperCase() + item.substr(1)
  BackgroundAudioManager.prototype[`on${name}`] = function (callback) {
    callbacks[item].push(callback)
  }
})

let backgroundAudioManager

export function getBackgroundAudioManager () {
  return backgroundAudioManager || (backgroundAudioManager = new BackgroundAudioManager())
165
}