15500cea7S猫头猫import produce from 'immer'; 25500cea7S猫头猫import ReactNativeTrackPlayer, { 35500cea7S猫头猫 Event, 45500cea7S猫头猫 State, 55500cea7S猫头猫 Track, 65500cea7S猫头猫 TrackMetadataBase, 75500cea7S猫头猫 usePlaybackState, 85500cea7S猫头猫 useProgress, 95500cea7S猫头猫} from 'react-native-track-player'; 105500cea7S猫头猫import shuffle from 'lodash.shuffle'; 115500cea7S猫头猫import Config from '../config'; 125500cea7S猫头猫import { 135500cea7S猫头猫 EDeviceEvents, 145500cea7S猫头猫 internalFakeSoundKey, 155500cea7S猫头猫 sortIndexSymbol, 165500cea7S猫头猫 timeStampSymbol, 175500cea7S猫头猫} from '@/constants/commonConst'; 185500cea7S猫头猫import {GlobalState} from '@/utils/stateMapper'; 195500cea7S猫头猫import delay from '@/utils/delay'; 205500cea7S猫头猫import { 215500cea7S猫头猫 isSameMediaItem, 225500cea7S猫头猫 mergeProps, 235500cea7S猫头猫 sortByTimestampAndIndex, 245500cea7S猫头猫} from '@/utils/mediaItem'; 255500cea7S猫头猫import Network from '../network'; 265500cea7S猫头猫import LocalMusicSheet from '../localMusicSheet'; 275500cea7S猫头猫import {SoundAsset} from '@/constants/assetsConst'; 285500cea7S猫头猫import {getQualityOrder} from '@/utils/qualities'; 295500cea7S猫头猫import musicHistory from '../musicHistory'; 305500cea7S猫头猫import getUrlExt from '@/utils/getUrlExt'; 315500cea7S猫头猫import {DeviceEventEmitter} from 'react-native'; 325500cea7S猫头猫import LyricManager from '../lyricManager'; 335500cea7S猫头猫import {MusicRepeatMode} from './common'; 345500cea7S猫头猫import { 355500cea7S猫头猫 getMusicIndex, 365500cea7S猫头猫 getPlayList, 375500cea7S猫头猫 getPlayListMusicAt, 385500cea7S猫头猫 isInPlayList, 395500cea7S猫头猫 isPlayListEmpty, 405500cea7S猫头猫 setPlayList, 415500cea7S猫头猫 usePlayList, 425500cea7S猫头猫} from './internal/playList'; 435500cea7S猫头猫import {createMediaIndexMap} from '@/utils/mediaIndexMap'; 445500cea7S猫头猫import PluginManager from '../pluginManager'; 455500cea7S猫头猫import {musicIsPaused} from '@/utils/trackUtils'; 4662e73a5eS猫头猫import {trace} from '@/utils/log'; 476e000b99S猫头猫import PersistStatus from '../persistStatus'; 48dffbbaffS猫头猫import {getCurrentDialog, showDialog} from '@/components/dialogs/useDialog'; 495500cea7S猫头猫 505500cea7S猫头猫/** 当前播放 */ 515500cea7S猫头猫const currentMusicStore = new GlobalState<IMusic.IMusicItem | null>(null); 525500cea7S猫头猫 535500cea7S猫头猫/** 播放模式 */ 545500cea7S猫头猫const repeatModeStore = new GlobalState<MusicRepeatMode>(MusicRepeatMode.QUEUE); 555500cea7S猫头猫 565500cea7S猫头猫/** 音质 */ 575500cea7S猫头猫const qualityStore = new GlobalState<IMusic.IQualityKey>('standard'); 585500cea7S猫头猫 595500cea7S猫头猫let currentIndex = -1; 605500cea7S猫头猫 615500cea7S猫头猫// TODO: 下个版本最大限制调大一些 625500cea7S猫头猫const maxMusicQueueLength = 1500; // 当前播放最大限制 635500cea7S猫头猫const halfMaxMusicQueueLength = Math.floor(maxMusicQueueLength / 2); 645500cea7S猫头猫const shrinkPlayListToSize = ( 655500cea7S猫头猫 queue: IMusic.IMusicItem[], 665500cea7S猫头猫 targetIndex = currentIndex, 675500cea7S猫头猫) => { 685500cea7S猫头猫 // 播放列表上限,太多无法缓存状态 695500cea7S猫头猫 if (queue.length > maxMusicQueueLength) { 705500cea7S猫头猫 if (targetIndex < halfMaxMusicQueueLength) { 715500cea7S猫头猫 queue = queue.slice(0, maxMusicQueueLength); 725500cea7S猫头猫 } else { 735500cea7S猫头猫 const right = Math.min( 745500cea7S猫头猫 queue.length, 755500cea7S猫头猫 targetIndex + halfMaxMusicQueueLength, 765500cea7S猫头猫 ); 775500cea7S猫头猫 const left = Math.max(0, right - maxMusicQueueLength); 785500cea7S猫头猫 queue = queue.slice(left, right); 795500cea7S猫头猫 } 805500cea7S猫头猫 } 815500cea7S猫头猫 return queue; 825500cea7S猫头猫}; 835500cea7S猫头猫 847aed04d4S猫头猫let hasSetupListener = false; 857aed04d4S猫头猫 866e000b99S猫头猫// TODO: 删除 876e000b99S猫头猫function migrate() { 886e000b99S猫头猫 const config = Config.get('status.music'); 896e000b99S猫头猫 if (!config) { 906e000b99S猫头猫 return; 916e000b99S猫头猫 } 925500cea7S猫头猫 const {rate, repeatMode, musicQueue, progress, track} = config; 936e000b99S猫头猫 PersistStatus.set('music.rate', rate); 946e000b99S猫头猫 PersistStatus.set('music.repeatMode', repeatMode); 956e000b99S猫头猫 PersistStatus.set('music.playList', musicQueue); 966e000b99S猫头猫 PersistStatus.set('music.progress', progress); 976e000b99S猫头猫 PersistStatus.set('music.musicItem', track); 986e000b99S猫头猫 Config.set('status.music', undefined); 996e000b99S猫头猫} 1006e000b99S猫头猫 1016e000b99S猫头猫async function setupTrackPlayer() { 1026e000b99S猫头猫 migrate(); 1036e000b99S猫头猫 1046e000b99S猫头猫 const rate = PersistStatus.get('music.rate'); 1056e000b99S猫头猫 const musicQueue = PersistStatus.get('music.playList'); 1066e000b99S猫头猫 const repeatMode = PersistStatus.get('music.repeatMode'); 1076e000b99S猫头猫 const progress = PersistStatus.get('music.progress'); 1086e000b99S猫头猫 const track = PersistStatus.get('music.musicItem'); 1096e000b99S猫头猫 const quality = 1106e000b99S猫头猫 PersistStatus.get('music.quality') || 1116e000b99S猫头猫 Config.get('setting.basic.defaultPlayQuality') || 1126e000b99S猫头猫 'standard'; 1135500cea7S猫头猫 1145500cea7S猫头猫 // 状态恢复 1155500cea7S猫头猫 if (rate) { 11691eb8fa8S猫头猫 ReactNativeTrackPlayer.setRate(+rate / 100); 1175500cea7S猫头猫 } 118e476f1a0S猫头猫 if (repeatMode) { 119e476f1a0S猫头猫 repeatModeStore.setValue(repeatMode as MusicRepeatMode); 120e476f1a0S猫头猫 } 1215500cea7S猫头猫 1225500cea7S猫头猫 if (musicQueue && Array.isArray(musicQueue)) { 1235500cea7S猫头猫 addAll(musicQueue, undefined, repeatMode === MusicRepeatMode.SHUFFLE); 1245500cea7S猫头猫 } 1255500cea7S猫头猫 1265500cea7S猫头猫 if (track && isInPlayList(track)) { 1278b9904b3S猫头猫 if (!Config.get('setting.basic.autoPlayWhenAppStart')) { 1288b9904b3S猫头猫 track.isInit = true; 1298b9904b3S猫头猫 } 1300dd58d38S猫头猫 13191eb8fa8S猫头猫 // 异步 13291eb8fa8S猫头猫 PluginManager.getByMedia(track) 13391eb8fa8S猫头猫 ?.methods.getMediaSource(track, quality, 0) 13491eb8fa8S猫头猫 .then(async newSource => { 13591eb8fa8S猫头猫 track.url = newSource?.url || track.url; 13691eb8fa8S猫头猫 track.headers = newSource?.headers || track.headers; 13791eb8fa8S猫头猫 13891eb8fa8S猫头猫 if (isSameMediaItem(currentMusicStore.getValue(), track)) { 1395500cea7S猫头猫 await setTrackSource(track as Track, false); 14091eb8fa8S猫头猫 } 14191eb8fa8S猫头猫 }); 1425500cea7S猫头猫 setCurrentMusic(track); 1435500cea7S猫头猫 1446e000b99S猫头猫 if (progress) { 14591eb8fa8S猫头猫 // 异步 14691eb8fa8S猫头猫 ReactNativeTrackPlayer.seekTo(progress); 1475500cea7S猫头猫 } 1485500cea7S猫头猫 } 1495500cea7S猫头猫 1507aed04d4S猫头猫 if (!hasSetupListener) { 1515500cea7S猫头猫 ReactNativeTrackPlayer.addEventListener( 1525500cea7S猫头猫 Event.PlaybackActiveTrackChanged, 1535500cea7S猫头猫 async evt => { 1545500cea7S猫头猫 if ( 1555500cea7S猫头猫 evt.index === 1 && 1565500cea7S猫头猫 evt.lastIndex === 0 && 1575500cea7S猫头猫 evt.track?.$ === internalFakeSoundKey 1585500cea7S猫头猫 ) { 15962e73a5eS猫头猫 trace('队列末尾,播放下一首'); 1605500cea7S猫头猫 if (repeatModeStore.getValue() === MusicRepeatMode.SINGLE) { 1615500cea7S猫头猫 await play(null, true); 1625500cea7S猫头猫 } else { 1635500cea7S猫头猫 // 当前生效的歌曲是下一曲的标记 1646f57784cS猫头猫 await skipToNext(); 1655500cea7S猫头猫 } 1665500cea7S猫头猫 } 1675500cea7S猫头猫 }, 1685500cea7S猫头猫 ); 1695500cea7S猫头猫 1707aed04d4S猫头猫 ReactNativeTrackPlayer.addEventListener( 1717aed04d4S猫头猫 Event.PlaybackError, 17262e73a5eS猫头猫 async e => { 17362e73a5eS猫头猫 // WARNING: 不稳定,报错的时候有可能track已经变到下一首歌去了 1748b9904b3S猫头猫 const currentTrack = 1758b9904b3S猫头猫 await ReactNativeTrackPlayer.getActiveTrack(); 1768b9904b3S猫头猫 if (currentTrack?.isInit) { 1778b9904b3S猫头猫 // HACK: 避免初始失败的情况 1788b9904b3S猫头猫 ReactNativeTrackPlayer.updateMetadataForTrack(0, { 1798b9904b3S猫头猫 ...currentTrack, 1808b9904b3S猫头猫 // @ts-ignore 1818b9904b3S猫头猫 isInit: undefined, 1828b9904b3S猫头猫 }); 1838b9904b3S猫头猫 return; 1848b9904b3S猫头猫 } 1858b9904b3S猫头猫 1867aed04d4S猫头猫 if ( 18762e73a5eS猫头猫 (await ReactNativeTrackPlayer.getActiveTrackIndex()) === 18862e73a5eS猫头猫 0 && 18962e73a5eS猫头猫 e.message && 19062e73a5eS猫头猫 e.message !== 'android-io-file-not-found' 1917aed04d4S猫头猫 ) { 19262e73a5eS猫头猫 trace('播放出错', { 19362e73a5eS猫头猫 message: e.message, 19462e73a5eS猫头猫 code: e.code, 19562e73a5eS猫头猫 }); 1968b9904b3S猫头猫 1975500cea7S猫头猫 failToPlay(); 1985500cea7S猫头猫 } 1997aed04d4S猫头猫 }, 2007aed04d4S猫头猫 ); 2017aed04d4S猫头猫 2027aed04d4S猫头猫 hasSetupListener = true; 2037aed04d4S猫头猫 } 2045500cea7S猫头猫} 2055500cea7S猫头猫 2065500cea7S猫头猫/** 2075500cea7S猫头猫 * 获取自动播放的下一个track 2085500cea7S猫头猫 */ 2095500cea7S猫头猫const getFakeNextTrack = () => { 2105500cea7S猫头猫 let track: Track | undefined; 2115500cea7S猫头猫 const repeatMode = repeatModeStore.getValue(); 2125500cea7S猫头猫 if (repeatMode === MusicRepeatMode.SINGLE) { 2135500cea7S猫头猫 // 单曲循环 2145500cea7S猫头猫 track = getPlayListMusicAt(currentIndex) as Track; 2155500cea7S猫头猫 } else { 2165500cea7S猫头猫 // 下一曲 2175500cea7S猫头猫 track = getPlayListMusicAt(currentIndex + 1) as Track; 2185500cea7S猫头猫 } 2195500cea7S猫头猫 2205500cea7S猫头猫 if (track) { 2215500cea7S猫头猫 return produce(track, _ => { 2225500cea7S猫头猫 _.url = SoundAsset.fakeAudio; 2235500cea7S猫头猫 _.$ = internalFakeSoundKey; 224cc77e86bS猫头猫 if (!_.artwork?.trim()?.length) { 225cc77e86bS猫头猫 _.artwork = undefined; 226cc77e86bS猫头猫 } 2275500cea7S猫头猫 }); 2285500cea7S猫头猫 } else { 2295500cea7S猫头猫 // 只有列表长度为0时才会出现的特殊情况 2305500cea7S猫头猫 return {url: SoundAsset.fakeAudio, $: internalFakeSoundKey} as Track; 2315500cea7S猫头猫 } 2325500cea7S猫头猫}; 2335500cea7S猫头猫 2345500cea7S猫头猫/** 播放失败时的情况 */ 2356f57784cS猫头猫async function failToPlay() { 2365500cea7S猫头猫 // 如果自动跳转下一曲, 500s后自动跳转 2375500cea7S猫头猫 if (!Config.get('setting.basic.autoStopWhenError')) { 2385500cea7S猫头猫 await ReactNativeTrackPlayer.reset(); 2395500cea7S猫头猫 await delay(500); 2406f57784cS猫头猫 await skipToNext(); 2415500cea7S猫头猫 } 2425500cea7S猫头猫} 2435500cea7S猫头猫 2445500cea7S猫头猫// 播放模式相关 2455500cea7S猫头猫const _toggleRepeatMapping = { 2465500cea7S猫头猫 [MusicRepeatMode.SHUFFLE]: MusicRepeatMode.SINGLE, 2475500cea7S猫头猫 [MusicRepeatMode.SINGLE]: MusicRepeatMode.QUEUE, 2485500cea7S猫头猫 [MusicRepeatMode.QUEUE]: MusicRepeatMode.SHUFFLE, 2495500cea7S猫头猫}; 2505500cea7S猫头猫/** 切换下一个模式 */ 2515500cea7S猫头猫const toggleRepeatMode = () => { 2525500cea7S猫头猫 setRepeatMode(_toggleRepeatMapping[repeatModeStore.getValue()]); 2535500cea7S猫头猫}; 2545500cea7S猫头猫 2555500cea7S猫头猫/** 2565500cea7S猫头猫 * 添加到播放列表 2575500cea7S猫头猫 * @param musicItems 目标歌曲 2585500cea7S猫头猫 * @param beforeIndex 在第x首歌曲前添加 2595500cea7S猫头猫 * @param shouldShuffle 随机排序 2605500cea7S猫头猫 */ 2615500cea7S猫头猫const addAll = ( 2625500cea7S猫头猫 musicItems: Array<IMusic.IMusicItem> = [], 2635500cea7S猫头猫 beforeIndex?: number, 2645500cea7S猫头猫 shouldShuffle?: boolean, 2655500cea7S猫头猫) => { 2665500cea7S猫头猫 const now = Date.now(); 2675500cea7S猫头猫 let newPlayList: IMusic.IMusicItem[] = []; 2685500cea7S猫头猫 let currentPlayList = getPlayList(); 2695500cea7S猫头猫 const _musicItems = musicItems.map((item, index) => 2705500cea7S猫头猫 produce(item, draft => { 2715500cea7S猫头猫 draft[timeStampSymbol] = now; 2725500cea7S猫头猫 draft[sortIndexSymbol] = index; 2735500cea7S猫头猫 }), 2745500cea7S猫头猫 ); 2755500cea7S猫头猫 if (beforeIndex === undefined || beforeIndex < 0) { 2765500cea7S猫头猫 // 1.1. 添加到歌单末尾,并过滤掉已有的歌曲 2775500cea7S猫头猫 newPlayList = currentPlayList.concat( 2785500cea7S猫头猫 _musicItems.filter(item => !isInPlayList(item)), 2795500cea7S猫头猫 ); 2805500cea7S猫头猫 } else { 2815500cea7S猫头猫 // 1.2. 新的播放列表,插入 2825500cea7S猫头猫 const indexMap = createMediaIndexMap(_musicItems); 2835500cea7S猫头猫 const beforeDraft = currentPlayList 2845500cea7S猫头猫 .slice(0, beforeIndex) 2855500cea7S猫头猫 .filter(item => !indexMap.has(item)); 2865500cea7S猫头猫 const afterDraft = currentPlayList 2875500cea7S猫头猫 .slice(beforeIndex) 2885500cea7S猫头猫 .filter(item => !indexMap.has(item)); 2895500cea7S猫头猫 2905500cea7S猫头猫 newPlayList = [...beforeDraft, ..._musicItems, ...afterDraft]; 2915500cea7S猫头猫 } 29215900d05S猫头猫 2935500cea7S猫头猫 // 如果太长了 2945500cea7S猫头猫 if (newPlayList.length > maxMusicQueueLength) { 2955500cea7S猫头猫 newPlayList = shrinkPlayListToSize( 2965500cea7S猫头猫 newPlayList, 2975500cea7S猫头猫 beforeIndex ?? newPlayList.length - 1, 2985500cea7S猫头猫 ); 2995500cea7S猫头猫 } 3005500cea7S猫头猫 3015500cea7S猫头猫 // 2. 如果需要随机 3025500cea7S猫头猫 if (shouldShuffle) { 3035500cea7S猫头猫 newPlayList = shuffle(newPlayList); 3045500cea7S猫头猫 } 3055500cea7S猫头猫 // 3. 设置播放列表 3065500cea7S猫头猫 setPlayList(newPlayList); 3075500cea7S猫头猫 const currentMusicItem = currentMusicStore.getValue(); 3085500cea7S猫头猫 3095500cea7S猫头猫 // 4. 重置下标 3105500cea7S猫头猫 if (currentMusicItem) { 3115500cea7S猫头猫 currentIndex = getMusicIndex(currentMusicItem); 3125500cea7S猫头猫 } 3135500cea7S猫头猫 3145500cea7S猫头猫 // TODO: 更新播放队列信息 3155500cea7S猫头猫 // 5. 存储更新的播放列表信息 3165500cea7S猫头猫}; 3175500cea7S猫头猫 3185500cea7S猫头猫/** 追加到队尾 */ 3195500cea7S猫头猫const add = ( 3205500cea7S猫头猫 musicItem: IMusic.IMusicItem | IMusic.IMusicItem[], 3215500cea7S猫头猫 beforeIndex?: number, 3225500cea7S猫头猫) => { 3235500cea7S猫头猫 addAll(Array.isArray(musicItem) ? musicItem : [musicItem], beforeIndex); 3245500cea7S猫头猫}; 3255500cea7S猫头猫 3265500cea7S猫头猫/** 3275500cea7S猫头猫 * 下一首播放 3285500cea7S猫头猫 * @param musicItem 3295500cea7S猫头猫 */ 3305500cea7S猫头猫const addNext = (musicItem: IMusic.IMusicItem | IMusic.IMusicItem[]) => { 3315500cea7S猫头猫 const shouldPlay = isPlayListEmpty(); 3325500cea7S猫头猫 add(musicItem, currentIndex + 1); 3335500cea7S猫头猫 if (shouldPlay) { 3345500cea7S猫头猫 play(Array.isArray(musicItem) ? musicItem[0] : musicItem); 3355500cea7S猫头猫 } 3365500cea7S猫头猫}; 3375500cea7S猫头猫 3385500cea7S猫头猫const isCurrentMusic = (musicItem: IMusic.IMusicItem) => { 3395500cea7S猫头猫 return isSameMediaItem(musicItem, currentMusicStore.getValue()) ?? false; 3405500cea7S猫头猫}; 3415500cea7S猫头猫 3425500cea7S猫头猫const remove = async (musicItem: IMusic.IMusicItem) => { 3435500cea7S猫头猫 const playList = getPlayList(); 3445500cea7S猫头猫 let newPlayList: IMusic.IMusicItem[] = []; 3455500cea7S猫头猫 let currentMusic: IMusic.IMusicItem | null = currentMusicStore.getValue(); 3465500cea7S猫头猫 const targetIndex = getMusicIndex(musicItem); 3475500cea7S猫头猫 let shouldPlayCurrent: boolean | null = null; 3485500cea7S猫头猫 if (targetIndex === -1) { 3495500cea7S猫头猫 // 1. 这种情况应该是出错了 3505500cea7S猫头猫 return; 3515500cea7S猫头猫 } 3525500cea7S猫头猫 // 2. 移除的是当前项 3535500cea7S猫头猫 if (currentIndex === targetIndex) { 3545500cea7S猫头猫 // 2.1 停止播放,移除当前项 3555500cea7S猫头猫 newPlayList = produce(playList, draft => { 3565500cea7S猫头猫 draft.splice(targetIndex, 1); 3575500cea7S猫头猫 }); 3585500cea7S猫头猫 // 2.2 设置新的播放列表,并更新当前音乐 3595500cea7S猫头猫 if (newPlayList.length === 0) { 3605500cea7S猫头猫 currentMusic = null; 3615500cea7S猫头猫 shouldPlayCurrent = false; 3625500cea7S猫头猫 } else { 3635500cea7S猫头猫 currentMusic = newPlayList[currentIndex % newPlayList.length]; 3645500cea7S猫头猫 try { 3655500cea7S猫头猫 const state = (await ReactNativeTrackPlayer.getPlaybackState()) 3665500cea7S猫头猫 .state; 3675500cea7S猫头猫 if (musicIsPaused(state)) { 3685500cea7S猫头猫 shouldPlayCurrent = false; 3695500cea7S猫头猫 } else { 3705500cea7S猫头猫 shouldPlayCurrent = true; 3715500cea7S猫头猫 } 3725500cea7S猫头猫 } catch { 3735500cea7S猫头猫 shouldPlayCurrent = false; 3745500cea7S猫头猫 } 3755500cea7S猫头猫 } 3765500cea7S猫头猫 } else { 3775500cea7S猫头猫 // 3. 删除 3785500cea7S猫头猫 newPlayList = produce(playList, draft => { 3795500cea7S猫头猫 draft.splice(targetIndex, 1); 3805500cea7S猫头猫 }); 3815500cea7S猫头猫 } 3825500cea7S猫头猫 3835500cea7S猫头猫 setPlayList(newPlayList); 3845500cea7S猫头猫 setCurrentMusic(currentMusic); 3855500cea7S猫头猫 if (shouldPlayCurrent === true) { 3865500cea7S猫头猫 await play(currentMusic, true); 3875500cea7S猫头猫 } else if (shouldPlayCurrent === false) { 3885500cea7S猫头猫 await ReactNativeTrackPlayer.reset(); 3895500cea7S猫头猫 } 3905500cea7S猫头猫}; 3915500cea7S猫头猫 3925500cea7S猫头猫/** 3935500cea7S猫头猫 * 设置播放模式 3945500cea7S猫头猫 * @param mode 播放模式 3955500cea7S猫头猫 */ 3965500cea7S猫头猫const setRepeatMode = (mode: MusicRepeatMode) => { 3975500cea7S猫头猫 const playList = getPlayList(); 3985500cea7S猫头猫 let newPlayList; 39991eb8fa8S猫头猫 const prevMode = repeatModeStore.getValue(); 40091eb8fa8S猫头猫 40191eb8fa8S猫头猫 if ( 40291eb8fa8S猫头猫 (prevMode === MusicRepeatMode.SHUFFLE && 40391eb8fa8S猫头猫 mode !== MusicRepeatMode.SHUFFLE) || 40491eb8fa8S猫头猫 (mode === MusicRepeatMode.SHUFFLE && 40591eb8fa8S猫头猫 prevMode !== MusicRepeatMode.SHUFFLE) 40691eb8fa8S猫头猫 ) { 4075500cea7S猫头猫 if (mode === MusicRepeatMode.SHUFFLE) { 4085500cea7S猫头猫 newPlayList = shuffle(playList); 4095500cea7S猫头猫 } else { 41091eb8fa8S猫头猫 newPlayList = sortByTimestampAndIndex(playList, true); 41191eb8fa8S猫头猫 } 41291eb8fa8S猫头猫 setPlayList(newPlayList); 4135500cea7S猫头猫 } 4145500cea7S猫头猫 4155500cea7S猫头猫 const currentMusicItem = currentMusicStore.getValue(); 4165500cea7S猫头猫 currentIndex = getMusicIndex(currentMusicItem); 4175500cea7S猫头猫 repeatModeStore.setValue(mode); 4185500cea7S猫头猫 // 更新下一首歌的信息 4195500cea7S猫头猫 ReactNativeTrackPlayer.updateMetadataForTrack(1, getFakeNextTrack()); 4205500cea7S猫头猫 // 记录 4216e000b99S猫头猫 PersistStatus.set('music.repeatMode', mode); 4225500cea7S猫头猫}; 4235500cea7S猫头猫 4245500cea7S猫头猫/** 清空播放列表 */ 4255500cea7S猫头猫const clear = async () => { 4265500cea7S猫头猫 setPlayList([]); 4275500cea7S猫头猫 setCurrentMusic(null); 4285500cea7S猫头猫 4295500cea7S猫头猫 await ReactNativeTrackPlayer.reset(); 4306e000b99S猫头猫 PersistStatus.set('music.musicItem', undefined); 4316e000b99S猫头猫 PersistStatus.set('music.progress', 0); 4325500cea7S猫头猫}; 4335500cea7S猫头猫 4345500cea7S猫头猫/** 暂停 */ 4355500cea7S猫头猫const pause = async () => { 4365500cea7S猫头猫 await ReactNativeTrackPlayer.pause(); 4375500cea7S猫头猫}; 4385500cea7S猫头猫 4398b9904b3S猫头猫/** 设置音源 */ 4408b9904b3S猫头猫const setTrackSource = async (track: Track, autoPlay = true) => { 44111908939S猫头猫 if (!track.artwork?.trim()?.length) { 44211908939S猫头猫 track.artwork = undefined; 44311908939S猫头猫 } 4448b9904b3S猫头猫 await ReactNativeTrackPlayer.setQueue([track, getFakeNextTrack()]); 4458b9904b3S猫头猫 PersistStatus.set('music.musicItem', track as IMusic.IMusicItem); 4468b9904b3S猫头猫 PersistStatus.set('music.progress', 0); 4478b9904b3S猫头猫 if (autoPlay) { 4488b9904b3S猫头猫 await ReactNativeTrackPlayer.play(); 4498b9904b3S猫头猫 } 4508b9904b3S猫头猫}; 4518b9904b3S猫头猫 4525500cea7S猫头猫const setCurrentMusic = (musicItem?: IMusic.IMusicItem | null) => { 4535500cea7S猫头猫 if (!musicItem) { 4545500cea7S猫头猫 currentIndex = -1; 455f511aee9S猫头猫 currentMusicStore.setValue(null); 4566e000b99S猫头猫 PersistStatus.set('music.musicItem', undefined); 4576e000b99S猫头猫 PersistStatus.set('music.progress', 0); 4586e000b99S猫头猫 return; 4595500cea7S猫头猫 } 4605500cea7S猫头猫 currentIndex = getMusicIndex(musicItem); 4616e000b99S猫头猫 currentMusicStore.setValue(musicItem); 4625500cea7S猫头猫}; 4635500cea7S猫头猫 4646e000b99S猫头猫const setQuality = (quality: IMusic.IQualityKey) => { 4656e000b99S猫头猫 qualityStore.setValue(quality); 4666e000b99S猫头猫 PersistStatus.set('music.quality', quality); 4676e000b99S猫头猫}; 4685500cea7S猫头猫/** 4695500cea7S猫头猫 * 播放 4705500cea7S猫头猫 * 4715500cea7S猫头猫 * 当musicItem 为空时,代表暂停/播放 4725500cea7S猫头猫 * 4735500cea7S猫头猫 * @param musicItem 4745500cea7S猫头猫 * @param forcePlay 4755500cea7S猫头猫 * @returns 4765500cea7S猫头猫 */ 4775500cea7S猫头猫const play = async ( 4785500cea7S猫头猫 musicItem?: IMusic.IMusicItem | null, 4795500cea7S猫头猫 forcePlay?: boolean, 4805500cea7S猫头猫) => { 4815500cea7S猫头猫 try { 4825500cea7S猫头猫 if (!musicItem) { 4835500cea7S猫头猫 musicItem = currentMusicStore.getValue(); 4845500cea7S猫头猫 } 4855500cea7S猫头猫 if (!musicItem) { 4865500cea7S猫头猫 throw new Error(PlayFailReason.PLAY_LIST_IS_EMPTY); 4875500cea7S猫头猫 } 4885500cea7S猫头猫 // 1. 移动网络禁止播放 4895500cea7S猫头猫 if ( 4905500cea7S猫头猫 Network.isCellular() && 4915500cea7S猫头猫 !Config.get('setting.basic.useCelluarNetworkPlay') && 4925500cea7S猫头猫 !LocalMusicSheet.isLocalMusic(musicItem) 4935500cea7S猫头猫 ) { 4945500cea7S猫头猫 await ReactNativeTrackPlayer.reset(); 4955500cea7S猫头猫 throw new Error(PlayFailReason.FORBID_CELLUAR_NETWORK_PLAY); 4965500cea7S猫头猫 } 4975500cea7S猫头猫 4985500cea7S猫头猫 // 2. 如果是当前正在播放的音频 4995500cea7S猫头猫 if (isCurrentMusic(musicItem)) { 5005500cea7S猫头猫 const currentTrack = await ReactNativeTrackPlayer.getTrack(0); 5015500cea7S猫头猫 // 2.1 如果当前有源 5025500cea7S猫头猫 if ( 5035500cea7S猫头猫 currentTrack?.url && 5045500cea7S猫头猫 isSameMediaItem(musicItem, currentTrack as IMusic.IMusicItem) 5055500cea7S猫头猫 ) { 5065500cea7S猫头猫 const currentActiveIndex = 5075500cea7S猫头猫 await ReactNativeTrackPlayer.getActiveTrackIndex(); 5085500cea7S猫头猫 if (currentActiveIndex !== 0) { 5095500cea7S猫头猫 await ReactNativeTrackPlayer.skip(0); 5105500cea7S猫头猫 } 5115500cea7S猫头猫 if (forcePlay) { 5125500cea7S猫头猫 // 2.1.1 强制重新开始 5135500cea7S猫头猫 await ReactNativeTrackPlayer.seekTo(0); 5146f57784cS猫头猫 } 51566e1d5fcS猫头猫 const currentState = ( 51666e1d5fcS猫头猫 await ReactNativeTrackPlayer.getPlaybackState() 51766e1d5fcS猫头猫 ).state; 51866e1d5fcS猫头猫 if (currentState === State.Stopped) { 51966e1d5fcS猫头猫 await setTrackSource(currentTrack); 52066e1d5fcS猫头猫 } 52166e1d5fcS猫头猫 if (currentState !== State.Playing) { 5225500cea7S猫头猫 // 2.1.2 恢复播放 5235500cea7S猫头猫 await ReactNativeTrackPlayer.play(); 5245500cea7S猫头猫 } 5255500cea7S猫头猫 // 这种情况下,播放队列和当前歌曲都不需要变化 5265500cea7S猫头猫 return; 5275500cea7S猫头猫 } 5285500cea7S猫头猫 // 2.2 其他情况:重新获取源 5295500cea7S猫头猫 } 5305500cea7S猫头猫 5315500cea7S猫头猫 // 3. 如果没有在播放列表中,添加到队尾;同时更新列表状态 5325500cea7S猫头猫 const inPlayList = isInPlayList(musicItem); 5335500cea7S猫头猫 if (!inPlayList) { 5345500cea7S猫头猫 add(musicItem); 5355500cea7S猫头猫 } 5365500cea7S猫头猫 5375500cea7S猫头猫 // 4. 更新列表状态和当前音乐 5385500cea7S猫头猫 setCurrentMusic(musicItem); 5399cfce1b6S猫头猫 await ReactNativeTrackPlayer.reset(); 5409cfce1b6S猫头猫 5419cfce1b6S猫头猫 // 4.1 刷新歌词信息 5429cfce1b6S猫头猫 if ( 5439cfce1b6S猫头猫 !isSameMediaItem( 5449cfce1b6S猫头猫 LyricManager.getLyricState()?.lyricParser?.getCurrentMusicItem?.(), 5459cfce1b6S猫头猫 musicItem, 5469cfce1b6S猫头猫 ) 5479cfce1b6S猫头猫 ) { 5489cfce1b6S猫头猫 DeviceEventEmitter.emit(EDeviceEvents.REFRESH_LYRIC, true); 5499cfce1b6S猫头猫 } 5505500cea7S猫头猫 5515500cea7S猫头猫 // 5. 获取音源 5525500cea7S猫头猫 let track: IMusic.IMusicItem; 5535500cea7S猫头猫 5545500cea7S猫头猫 // 5.1 通过插件获取音源 5555500cea7S猫头猫 const plugin = PluginManager.getByName(musicItem.platform); 5565500cea7S猫头猫 // 5.2 获取音质排序 5575500cea7S猫头猫 const qualityOrder = getQualityOrder( 5585500cea7S猫头猫 Config.get('setting.basic.defaultPlayQuality') ?? 'standard', 5595500cea7S猫头猫 Config.get('setting.basic.playQualityOrder') ?? 'asc', 5605500cea7S猫头猫 ); 5615500cea7S猫头猫 // 5.3 插件返回音源 5625500cea7S猫头猫 let source: IPlugin.IMediaSourceResult | null = null; 5635500cea7S猫头猫 for (let quality of qualityOrder) { 5645500cea7S猫头猫 if (isCurrentMusic(musicItem)) { 5655500cea7S猫头猫 source = 5665500cea7S猫头猫 (await plugin?.methods?.getMediaSource( 5675500cea7S猫头猫 musicItem, 5685500cea7S猫头猫 quality, 5695500cea7S猫头猫 )) ?? null; 5705500cea7S猫头猫 // 5.3.1 获取到真实源 5715500cea7S猫头猫 if (source) { 5726e000b99S猫头猫 setQuality(quality); 5735500cea7S猫头猫 break; 5745500cea7S猫头猫 } 5755500cea7S猫头猫 } else { 5765500cea7S猫头猫 // 5.3.2 已经切换到其他歌曲了, 5775500cea7S猫头猫 return; 5785500cea7S猫头猫 } 5795500cea7S猫头猫 } 5805500cea7S猫头猫 5815500cea7S猫头猫 if (!isCurrentMusic(musicItem)) { 5825500cea7S猫头猫 return; 5835500cea7S猫头猫 } 5845500cea7S猫头猫 5855500cea7S猫头猫 if (!source) { 58643eb30bfS猫头猫 // 如果有source 58743eb30bfS猫头猫 if (musicItem.source) { 58843eb30bfS猫头猫 for (let quality of qualityOrder) { 58943eb30bfS猫头猫 if (musicItem.source[quality]?.url) { 59043eb30bfS猫头猫 source = musicItem.source[quality]!; 5916e000b99S猫头猫 setQuality(quality); 5926e000b99S猫头猫 59343eb30bfS猫头猫 break; 5945500cea7S猫头猫 } 59543eb30bfS猫头猫 } 59643eb30bfS猫头猫 } 59743eb30bfS猫头猫 59843eb30bfS猫头猫 // 5.4 没有返回源 59943eb30bfS猫头猫 if (!source && !musicItem.url) { 60043eb30bfS猫头猫 throw new Error(PlayFailReason.INVALID_SOURCE); 60143eb30bfS猫头猫 } else { 6025500cea7S猫头猫 source = { 6035500cea7S猫头猫 url: musicItem.url, 6045500cea7S猫头猫 }; 6056e000b99S猫头猫 setQuality('standard'); 6065500cea7S猫头猫 } 60743eb30bfS猫头猫 } 6085500cea7S猫头猫 6095500cea7S猫头猫 // 6. 特殊类型源 6105500cea7S猫头猫 if (getUrlExt(source.url) === '.m3u8') { 6115500cea7S猫头猫 // @ts-ignore 6125500cea7S猫头猫 source.type = 'hls'; 6135500cea7S猫头猫 } 6145500cea7S猫头猫 // 7. 合并结果 6155500cea7S猫头猫 track = mergeProps(musicItem, source) as IMusic.IMusicItem; 6165500cea7S猫头猫 6175500cea7S猫头猫 // 8. 新增历史记录 6185500cea7S猫头猫 musicHistory.addMusic(musicItem); 6195500cea7S猫头猫 62062e73a5eS猫头猫 trace('获取音源成功', track); 6215500cea7S猫头猫 // 9. 设置音源 6225500cea7S猫头猫 await setTrackSource(track as Track); 6235500cea7S猫头猫 6245500cea7S猫头猫 // 10. 获取补充信息 6255500cea7S猫头猫 let info: Partial<IMusic.IMusicItem> | null = null; 6265500cea7S猫头猫 try { 6275500cea7S猫头猫 info = (await plugin?.methods?.getMusicInfo?.(musicItem)) ?? null; 62863d241f1S猫头猫 if ( 62963d241f1S猫头猫 (typeof info?.url === 'string' && info.url.trim() === '') || 63063d241f1S猫头猫 (info?.url && typeof info.url !== 'string') 63163d241f1S猫头猫 ) { 632f9c53a4cS猫头猫 delete info.url; 633f9c53a4cS猫头猫 } 6345500cea7S猫头猫 } catch {} 6355500cea7S猫头猫 6365500cea7S猫头猫 // 11. 设置补充信息 6375500cea7S猫头猫 if (info && isCurrentMusic(musicItem)) { 6385500cea7S猫头猫 const mergedTrack = mergeProps(track, info); 6395500cea7S猫头猫 currentMusicStore.setValue(mergedTrack as IMusic.IMusicItem); 6405500cea7S猫头猫 await ReactNativeTrackPlayer.updateMetadataForTrack( 6415500cea7S猫头猫 0, 6425500cea7S猫头猫 mergedTrack as TrackMetadataBase, 6435500cea7S猫头猫 ); 6445500cea7S猫头猫 } 6455500cea7S猫头猫 } catch (e: any) { 6465500cea7S猫头猫 const message = e?.message; 6475500cea7S猫头猫 if ( 6485500cea7S猫头猫 message === 'The player is not initialized. Call setupPlayer first.' 6495500cea7S猫头猫 ) { 6505500cea7S猫头猫 await ReactNativeTrackPlayer.setupPlayer(); 6515500cea7S猫头猫 play(musicItem, forcePlay); 6525500cea7S猫头猫 } else if (message === PlayFailReason.FORBID_CELLUAR_NETWORK_PLAY) { 653dffbbaffS猫头猫 if (getCurrentDialog()?.name !== 'SimpleDialog') { 654595a35deS猫头猫 showDialog('SimpleDialog', { 655595a35deS猫头猫 title: '流量提醒', 656595a35deS猫头猫 content: 657*2dd57e49S猫头猫 '当前非WIFI环境,侧边栏设置中打开【使用移动网络播放】功能后可继续播放', 658595a35deS猫头猫 }); 659dffbbaffS猫头猫 } 660dffbbaffS猫头猫 661dffbbaffS猫头猫 // Toast.warn( 662dffbbaffS猫头猫 // '当前禁止移动网络播放音乐,如需播放请去侧边栏-基本设置中修改', 663dffbbaffS猫头猫 // ); 6645500cea7S猫头猫 } else if (message === PlayFailReason.INVALID_SOURCE) { 66562e73a5eS猫头猫 trace('音源为空,播放失败'); 6666f57784cS猫头猫 await failToPlay(); 6675500cea7S猫头猫 } else if (message === PlayFailReason.PLAY_LIST_IS_EMPTY) { 6685500cea7S猫头猫 // 队列是空的,不应该出现这种情况 6695500cea7S猫头猫 } 6705500cea7S猫头猫 } 6715500cea7S猫头猫}; 6725500cea7S猫头猫 6735500cea7S猫头猫/** 6745500cea7S猫头猫 * 播放音乐,同时替换播放队列 6755500cea7S猫头猫 * @param musicItem 音乐 6765500cea7S猫头猫 * @param newPlayList 替代列表 6775500cea7S猫头猫 */ 6785500cea7S猫头猫const playWithReplacePlayList = async ( 6795500cea7S猫头猫 musicItem: IMusic.IMusicItem, 6805500cea7S猫头猫 newPlayList: IMusic.IMusicItem[], 6815500cea7S猫头猫) => { 6825500cea7S猫头猫 if (newPlayList.length !== 0) { 6835500cea7S猫头猫 const now = Date.now(); 6845500cea7S猫头猫 if (newPlayList.length > maxMusicQueueLength) { 6855500cea7S猫头猫 newPlayList = shrinkPlayListToSize( 6865500cea7S猫头猫 newPlayList, 6875500cea7S猫头猫 newPlayList.findIndex(it => isSameMediaItem(it, musicItem)), 6885500cea7S猫头猫 ); 6895500cea7S猫头猫 } 6905500cea7S猫头猫 const playListItems = newPlayList.map((item, index) => 6915500cea7S猫头猫 produce(item, draft => { 6925500cea7S猫头猫 draft[timeStampSymbol] = now; 6935500cea7S猫头猫 draft[sortIndexSymbol] = index; 6945500cea7S猫头猫 }), 6955500cea7S猫头猫 ); 6965500cea7S猫头猫 setPlayList( 6975500cea7S猫头猫 repeatModeStore.getValue() === MusicRepeatMode.SHUFFLE 6985500cea7S猫头猫 ? shuffle(playListItems) 6995500cea7S猫头猫 : playListItems, 7005500cea7S猫头猫 ); 7015500cea7S猫头猫 await play(musicItem, true); 7025500cea7S猫头猫 } 7035500cea7S猫头猫}; 7045500cea7S猫头猫 7056f57784cS猫头猫const skipToNext = async () => { 7065500cea7S猫头猫 if (isPlayListEmpty()) { 7075500cea7S猫头猫 setCurrentMusic(null); 7085500cea7S猫头猫 return; 7095500cea7S猫头猫 } 7105500cea7S猫头猫 7115500cea7S猫头猫 await play(getPlayListMusicAt(currentIndex + 1), true); 7125500cea7S猫头猫}; 7135500cea7S猫头猫 7145500cea7S猫头猫const skipToPrevious = async () => { 7155500cea7S猫头猫 if (isPlayListEmpty()) { 7165500cea7S猫头猫 setCurrentMusic(null); 7175500cea7S猫头猫 return; 7185500cea7S猫头猫 } 7195500cea7S猫头猫 7206f57784cS猫头猫 await play( 7216f57784cS猫头猫 getPlayListMusicAt(currentIndex === -1 ? 0 : currentIndex - 1), 7226f57784cS猫头猫 true, 7236f57784cS猫头猫 ); 7245500cea7S猫头猫}; 7255500cea7S猫头猫 7265500cea7S猫头猫/** 修改当前播放的音质 */ 7275500cea7S猫头猫const changeQuality = async (newQuality: IMusic.IQualityKey) => { 7285500cea7S猫头猫 // 获取当前的音乐和进度 7295500cea7S猫头猫 if (newQuality === qualityStore.getValue()) { 7305500cea7S猫头猫 return true; 7315500cea7S猫头猫 } 7325500cea7S猫头猫 7335500cea7S猫头猫 // 获取当前歌曲 7345500cea7S猫头猫 const musicItem = currentMusicStore.getValue(); 7355500cea7S猫头猫 if (!musicItem) { 7365500cea7S猫头猫 return false; 7375500cea7S猫头猫 } 7385500cea7S猫头猫 try { 7395500cea7S猫头猫 const progress = await ReactNativeTrackPlayer.getProgress(); 7405500cea7S猫头猫 const plugin = PluginManager.getByMedia(musicItem); 7415500cea7S猫头猫 const newSource = await plugin?.methods?.getMediaSource( 7425500cea7S猫头猫 musicItem, 7435500cea7S猫头猫 newQuality, 7445500cea7S猫头猫 ); 7455500cea7S猫头猫 if (!newSource?.url) { 7465500cea7S猫头猫 throw new Error(PlayFailReason.INVALID_SOURCE); 7475500cea7S猫头猫 } 7485500cea7S猫头猫 if (isCurrentMusic(musicItem)) { 7495500cea7S猫头猫 const playingState = ( 7505500cea7S猫头猫 await ReactNativeTrackPlayer.getPlaybackState() 7515500cea7S猫头猫 ).state; 7525500cea7S猫头猫 await setTrackSource( 7535500cea7S猫头猫 mergeProps(musicItem, newSource) as unknown as Track, 7545500cea7S猫头猫 !musicIsPaused(playingState), 7555500cea7S猫头猫 ); 7565500cea7S猫头猫 7575500cea7S猫头猫 await ReactNativeTrackPlayer.seekTo(progress.position ?? 0); 7586e000b99S猫头猫 setQuality(newQuality); 7595500cea7S猫头猫 } 7605500cea7S猫头猫 return true; 7615500cea7S猫头猫 } catch { 7625500cea7S猫头猫 // 修改失败 7635500cea7S猫头猫 return false; 7645500cea7S猫头猫 } 7655500cea7S猫头猫}; 7665500cea7S猫头猫 7675500cea7S猫头猫enum PlayFailReason { 7685500cea7S猫头猫 /** 禁止移动网络播放 */ 7695500cea7S猫头猫 FORBID_CELLUAR_NETWORK_PLAY = 'FORBID_CELLUAR_NETWORK_PLAY', 7705500cea7S猫头猫 /** 播放列表为空 */ 7715500cea7S猫头猫 PLAY_LIST_IS_EMPTY = 'PLAY_LIST_IS_EMPTY', 7725500cea7S猫头猫 /** 无效源 */ 7735500cea7S猫头猫 INVALID_SOURCE = 'INVALID_SOURCE', 7745500cea7S猫头猫 /** 非当前音乐 */ 7755500cea7S猫头猫} 7765500cea7S猫头猫 7775500cea7S猫头猫function useMusicState() { 7785500cea7S猫头猫 const playbackState = usePlaybackState(); 7795500cea7S猫头猫 7805500cea7S猫头猫 return playbackState.state; 7815500cea7S猫头猫} 7825500cea7S猫头猫 783f511aee9S猫头猫function getPreviousMusic() { 784f511aee9S猫头猫 const currentMusicItem = currentMusicStore.getValue(); 785f511aee9S猫头猫 if (!currentMusicItem) { 786f511aee9S猫头猫 return null; 787f511aee9S猫头猫 } 788f511aee9S猫头猫 789f511aee9S猫头猫 return getPlayListMusicAt(currentIndex - 1); 790f511aee9S猫头猫} 791f511aee9S猫头猫 792f511aee9S猫头猫function getNextMusic() { 793f511aee9S猫头猫 const currentMusicItem = currentMusicStore.getValue(); 794f511aee9S猫头猫 if (!currentMusicItem) { 795f511aee9S猫头猫 return null; 796f511aee9S猫头猫 } 797f511aee9S猫头猫 798f511aee9S猫头猫 return getPlayListMusicAt(currentIndex + 1); 799f511aee9S猫头猫} 800f511aee9S猫头猫 8015500cea7S猫头猫const TrackPlayer = { 8025500cea7S猫头猫 setupTrackPlayer, 8035500cea7S猫头猫 usePlayList, 8045500cea7S猫头猫 getPlayList, 8055500cea7S猫头猫 addAll, 8065500cea7S猫头猫 add, 8075500cea7S猫头猫 addNext, 8085500cea7S猫头猫 skipToNext, 8095500cea7S猫头猫 skipToPrevious, 8105500cea7S猫头猫 play, 8115500cea7S猫头猫 playWithReplacePlayList, 8125500cea7S猫头猫 pause, 8135500cea7S猫头猫 remove, 8145500cea7S猫头猫 clear, 8155500cea7S猫头猫 useCurrentMusic: currentMusicStore.useValue, 8165500cea7S猫头猫 getCurrentMusic: currentMusicStore.getValue, 8175500cea7S猫头猫 useRepeatMode: repeatModeStore.useValue, 8185500cea7S猫头猫 getRepeatMode: repeatModeStore.getValue, 8195500cea7S猫头猫 toggleRepeatMode, 8205500cea7S猫头猫 usePlaybackState, 8215500cea7S猫头猫 getProgress: ReactNativeTrackPlayer.getProgress, 8225500cea7S猫头猫 useProgress: useProgress, 8235500cea7S猫头猫 seekTo: ReactNativeTrackPlayer.seekTo, 8245500cea7S猫头猫 changeQuality, 8255500cea7S猫头猫 useCurrentQuality: qualityStore.useValue, 8265500cea7S猫头猫 getCurrentQuality: qualityStore.getValue, 8275500cea7S猫头猫 getRate: ReactNativeTrackPlayer.getRate, 8285500cea7S猫头猫 setRate: ReactNativeTrackPlayer.setRate, 8295500cea7S猫头猫 useMusicState, 8305500cea7S猫头猫 reset: ReactNativeTrackPlayer.reset, 831f511aee9S猫头猫 getPreviousMusic, 832f511aee9S猫头猫 getNextMusic, 8335500cea7S猫头猫}; 8345500cea7S猫头猫 8355500cea7S猫头猫export default TrackPlayer; 8365500cea7S猫头猫export {MusicRepeatMode, State as MusicState}; 837