xref: /MusicFree/src/core/trackPlayer/index.ts (revision 6e000b9998608a65a4fb0f4cc1aee5067e172a2d)
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';
46f511aee9S猫头猫import Toast from '@/utils/toast';
4762e73a5eS猫头猫import {trace} from '@/utils/log';
48*6e000b99S猫头猫import PersistStatus from '../persistStatus';
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猫头猫
86*6e000b99S猫头猫// TODO: 删除
87*6e000b99S猫头猫function migrate() {
88*6e000b99S猫头猫    const config = Config.get('status.music');
89*6e000b99S猫头猫    if (!config) {
90*6e000b99S猫头猫        return;
91*6e000b99S猫头猫    }
925500cea7S猫头猫    const {rate, repeatMode, musicQueue, progress, track} = config;
93*6e000b99S猫头猫    PersistStatus.set('music.rate', rate);
94*6e000b99S猫头猫    PersistStatus.set('music.repeatMode', repeatMode);
95*6e000b99S猫头猫    PersistStatus.set('music.playList', musicQueue);
96*6e000b99S猫头猫    PersistStatus.set('music.progress', progress);
97*6e000b99S猫头猫    PersistStatus.set('music.musicItem', track);
98*6e000b99S猫头猫    Config.set('status.music', undefined);
99*6e000b99S猫头猫}
100*6e000b99S猫头猫
101*6e000b99S猫头猫async function setupTrackPlayer() {
102*6e000b99S猫头猫    migrate();
103*6e000b99S猫头猫
104*6e000b99S猫头猫    const rate = PersistStatus.get('music.rate');
105*6e000b99S猫头猫    const musicQueue = PersistStatus.get('music.playList');
106*6e000b99S猫头猫    const repeatMode = PersistStatus.get('music.repeatMode');
107*6e000b99S猫头猫    const progress = PersistStatus.get('music.progress');
108*6e000b99S猫头猫    const track = PersistStatus.get('music.musicItem');
109*6e000b99S猫头猫    const quality =
110*6e000b99S猫头猫        PersistStatus.get('music.quality') ||
111*6e000b99S猫头猫        Config.get('setting.basic.defaultPlayQuality') ||
112*6e000b99S猫头猫        'standard';
1135500cea7S猫头猫
1145500cea7S猫头猫    // 状态恢复
1155500cea7S猫头猫    if (rate) {
1165500cea7S猫头猫        await ReactNativeTrackPlayer.setRate(+rate / 100);
1175500cea7S猫头猫    }
1185500cea7S猫头猫
1195500cea7S猫头猫    if (musicQueue && Array.isArray(musicQueue)) {
1205500cea7S猫头猫        addAll(musicQueue, undefined, repeatMode === MusicRepeatMode.SHUFFLE);
1215500cea7S猫头猫    }
1225500cea7S猫头猫
1235500cea7S猫头猫    if (track && isInPlayList(track)) {
1245500cea7S猫头猫        const newSource = await PluginManager.getByMedia(
1255500cea7S猫头猫            track,
126*6e000b99S猫头猫        )?.methods.getMediaSource(track, quality, 0);
1275500cea7S猫头猫        // 重新初始化 获取最新的链接
1285500cea7S猫头猫        track.url = newSource?.url || track.url;
1295500cea7S猫头猫        track.headers = newSource?.headers || track.headers;
1305500cea7S猫头猫
1315500cea7S猫头猫        await setTrackSource(track as Track, false);
1325500cea7S猫头猫        setCurrentMusic(track);
1335500cea7S猫头猫
134*6e000b99S猫头猫        if (progress) {
135*6e000b99S猫头猫            await ReactNativeTrackPlayer.seekTo(progress);
1365500cea7S猫头猫        }
1375500cea7S猫头猫    }
1385500cea7S猫头猫
1397aed04d4S猫头猫    if (!hasSetupListener) {
1405500cea7S猫头猫        ReactNativeTrackPlayer.addEventListener(
1415500cea7S猫头猫            Event.PlaybackActiveTrackChanged,
1425500cea7S猫头猫            async evt => {
1435500cea7S猫头猫                if (
1445500cea7S猫头猫                    evt.index === 1 &&
1455500cea7S猫头猫                    evt.lastIndex === 0 &&
1465500cea7S猫头猫                    evt.track?.$ === internalFakeSoundKey
1475500cea7S猫头猫                ) {
14862e73a5eS猫头猫                    trace('队列末尾,播放下一首');
1495500cea7S猫头猫                    if (repeatModeStore.getValue() === MusicRepeatMode.SINGLE) {
1505500cea7S猫头猫                        await play(null, true);
1515500cea7S猫头猫                    } else {
1525500cea7S猫头猫                        // 当前生效的歌曲是下一曲的标记
1536f57784cS猫头猫                        await skipToNext();
1545500cea7S猫头猫                    }
1555500cea7S猫头猫                }
1565500cea7S猫头猫            },
1575500cea7S猫头猫        );
1585500cea7S猫头猫
1597aed04d4S猫头猫        ReactNativeTrackPlayer.addEventListener(
1607aed04d4S猫头猫            Event.PlaybackError,
16162e73a5eS猫头猫            async e => {
16262e73a5eS猫头猫                // WARNING: 不稳定,报错的时候有可能track已经变到下一首歌去了
1637aed04d4S猫头猫                if (
16462e73a5eS猫头猫                    (await ReactNativeTrackPlayer.getActiveTrackIndex()) ===
16562e73a5eS猫头猫                        0 &&
16662e73a5eS猫头猫                    e.message &&
16762e73a5eS猫头猫                    e.message !== 'android-io-file-not-found'
1687aed04d4S猫头猫                ) {
16962e73a5eS猫头猫                    trace('播放出错', {
17062e73a5eS猫头猫                        message: e.message,
17162e73a5eS猫头猫                        code: e.code,
17262e73a5eS猫头猫                    });
1735500cea7S猫头猫                    failToPlay();
1745500cea7S猫头猫                }
1757aed04d4S猫头猫            },
1767aed04d4S猫头猫        );
1777aed04d4S猫头猫
1787aed04d4S猫头猫        hasSetupListener = true;
1797aed04d4S猫头猫    }
1805500cea7S猫头猫}
1815500cea7S猫头猫
1825500cea7S猫头猫/**
1835500cea7S猫头猫 * 获取自动播放的下一个track
1845500cea7S猫头猫 */
1855500cea7S猫头猫const getFakeNextTrack = () => {
1865500cea7S猫头猫    let track: Track | undefined;
1875500cea7S猫头猫    const repeatMode = repeatModeStore.getValue();
1885500cea7S猫头猫    if (repeatMode === MusicRepeatMode.SINGLE) {
1895500cea7S猫头猫        // 单曲循环
1905500cea7S猫头猫        track = getPlayListMusicAt(currentIndex) as Track;
1915500cea7S猫头猫    } else {
1925500cea7S猫头猫        // 下一曲
1935500cea7S猫头猫        track = getPlayListMusicAt(currentIndex + 1) as Track;
1945500cea7S猫头猫    }
1955500cea7S猫头猫
1965500cea7S猫头猫    if (track) {
1975500cea7S猫头猫        return produce(track, _ => {
1985500cea7S猫头猫            _.url = SoundAsset.fakeAudio;
1995500cea7S猫头猫            _.$ = internalFakeSoundKey;
2005500cea7S猫头猫        });
2015500cea7S猫头猫    } else {
2025500cea7S猫头猫        // 只有列表长度为0时才会出现的特殊情况
2035500cea7S猫头猫        return {url: SoundAsset.fakeAudio, $: internalFakeSoundKey} as Track;
2045500cea7S猫头猫    }
2055500cea7S猫头猫};
2065500cea7S猫头猫
2075500cea7S猫头猫/** 播放失败时的情况 */
2086f57784cS猫头猫async function failToPlay() {
2095500cea7S猫头猫    // 如果自动跳转下一曲, 500s后自动跳转
2105500cea7S猫头猫    if (!Config.get('setting.basic.autoStopWhenError')) {
2115500cea7S猫头猫        await ReactNativeTrackPlayer.reset();
2125500cea7S猫头猫        await delay(500);
2136f57784cS猫头猫        await skipToNext();
2145500cea7S猫头猫    }
2155500cea7S猫头猫}
2165500cea7S猫头猫
2175500cea7S猫头猫// 播放模式相关
2185500cea7S猫头猫const _toggleRepeatMapping = {
2195500cea7S猫头猫    [MusicRepeatMode.SHUFFLE]: MusicRepeatMode.SINGLE,
2205500cea7S猫头猫    [MusicRepeatMode.SINGLE]: MusicRepeatMode.QUEUE,
2215500cea7S猫头猫    [MusicRepeatMode.QUEUE]: MusicRepeatMode.SHUFFLE,
2225500cea7S猫头猫};
2235500cea7S猫头猫/** 切换下一个模式 */
2245500cea7S猫头猫const toggleRepeatMode = () => {
2255500cea7S猫头猫    setRepeatMode(_toggleRepeatMapping[repeatModeStore.getValue()]);
2265500cea7S猫头猫};
2275500cea7S猫头猫
2285500cea7S猫头猫/** 设置音源 */
2295500cea7S猫头猫const setTrackSource = async (track: Track, autoPlay = true) => {
2305500cea7S猫头猫    await ReactNativeTrackPlayer.setQueue([track, getFakeNextTrack()]);
2315500cea7S猫头猫    if (autoPlay) {
2325500cea7S猫头猫        await ReactNativeTrackPlayer.play();
2335500cea7S猫头猫    }
2345500cea7S猫头猫};
2355500cea7S猫头猫
2365500cea7S猫头猫/**
2375500cea7S猫头猫 * 添加到播放列表
2385500cea7S猫头猫 * @param musicItems 目标歌曲
2395500cea7S猫头猫 * @param beforeIndex 在第x首歌曲前添加
2405500cea7S猫头猫 * @param shouldShuffle 随机排序
2415500cea7S猫头猫 */
2425500cea7S猫头猫const addAll = (
2435500cea7S猫头猫    musicItems: Array<IMusic.IMusicItem> = [],
2445500cea7S猫头猫    beforeIndex?: number,
2455500cea7S猫头猫    shouldShuffle?: boolean,
2465500cea7S猫头猫) => {
2475500cea7S猫头猫    const now = Date.now();
2485500cea7S猫头猫    let newPlayList: IMusic.IMusicItem[] = [];
2495500cea7S猫头猫    let currentPlayList = getPlayList();
2505500cea7S猫头猫    const _musicItems = musicItems.map((item, index) =>
2515500cea7S猫头猫        produce(item, draft => {
2525500cea7S猫头猫            draft[timeStampSymbol] = now;
2535500cea7S猫头猫            draft[sortIndexSymbol] = index;
2545500cea7S猫头猫        }),
2555500cea7S猫头猫    );
2565500cea7S猫头猫    if (beforeIndex === undefined || beforeIndex < 0) {
2575500cea7S猫头猫        // 1.1. 添加到歌单末尾,并过滤掉已有的歌曲
2585500cea7S猫头猫        newPlayList = currentPlayList.concat(
2595500cea7S猫头猫            _musicItems.filter(item => !isInPlayList(item)),
2605500cea7S猫头猫        );
2615500cea7S猫头猫    } else {
2625500cea7S猫头猫        // 1.2. 新的播放列表,插入
2635500cea7S猫头猫        const indexMap = createMediaIndexMap(_musicItems);
2645500cea7S猫头猫        const beforeDraft = currentPlayList
2655500cea7S猫头猫            .slice(0, beforeIndex)
2665500cea7S猫头猫            .filter(item => !indexMap.has(item));
2675500cea7S猫头猫        const afterDraft = currentPlayList
2685500cea7S猫头猫            .slice(beforeIndex)
2695500cea7S猫头猫            .filter(item => !indexMap.has(item));
2705500cea7S猫头猫
2715500cea7S猫头猫        newPlayList = [...beforeDraft, ..._musicItems, ...afterDraft];
2725500cea7S猫头猫    }
27315900d05S猫头猫
2745500cea7S猫头猫    // 如果太长了
2755500cea7S猫头猫    if (newPlayList.length > maxMusicQueueLength) {
2765500cea7S猫头猫        newPlayList = shrinkPlayListToSize(
2775500cea7S猫头猫            newPlayList,
2785500cea7S猫头猫            beforeIndex ?? newPlayList.length - 1,
2795500cea7S猫头猫        );
2805500cea7S猫头猫    }
2815500cea7S猫头猫
2825500cea7S猫头猫    // 2. 如果需要随机
2835500cea7S猫头猫    if (shouldShuffle) {
2845500cea7S猫头猫        newPlayList = shuffle(newPlayList);
2855500cea7S猫头猫    }
2865500cea7S猫头猫    // 3. 设置播放列表
2875500cea7S猫头猫    setPlayList(newPlayList);
2885500cea7S猫头猫    const currentMusicItem = currentMusicStore.getValue();
2895500cea7S猫头猫
2905500cea7S猫头猫    // 4. 重置下标
2915500cea7S猫头猫    if (currentMusicItem) {
2925500cea7S猫头猫        currentIndex = getMusicIndex(currentMusicItem);
2935500cea7S猫头猫    }
2945500cea7S猫头猫
2955500cea7S猫头猫    // TODO: 更新播放队列信息
2965500cea7S猫头猫    // 5. 存储更新的播放列表信息
2975500cea7S猫头猫};
2985500cea7S猫头猫
2995500cea7S猫头猫/** 追加到队尾 */
3005500cea7S猫头猫const add = (
3015500cea7S猫头猫    musicItem: IMusic.IMusicItem | IMusic.IMusicItem[],
3025500cea7S猫头猫    beforeIndex?: number,
3035500cea7S猫头猫) => {
3045500cea7S猫头猫    addAll(Array.isArray(musicItem) ? musicItem : [musicItem], beforeIndex);
3055500cea7S猫头猫};
3065500cea7S猫头猫
3075500cea7S猫头猫/**
3085500cea7S猫头猫 * 下一首播放
3095500cea7S猫头猫 * @param musicItem
3105500cea7S猫头猫 */
3115500cea7S猫头猫const addNext = (musicItem: IMusic.IMusicItem | IMusic.IMusicItem[]) => {
3125500cea7S猫头猫    const shouldPlay = isPlayListEmpty();
3135500cea7S猫头猫    add(musicItem, currentIndex + 1);
3145500cea7S猫头猫    if (shouldPlay) {
3155500cea7S猫头猫        play(Array.isArray(musicItem) ? musicItem[0] : musicItem);
3165500cea7S猫头猫    }
3175500cea7S猫头猫};
3185500cea7S猫头猫
3195500cea7S猫头猫const isCurrentMusic = (musicItem: IMusic.IMusicItem) => {
3205500cea7S猫头猫    return isSameMediaItem(musicItem, currentMusicStore.getValue()) ?? false;
3215500cea7S猫头猫};
3225500cea7S猫头猫
3235500cea7S猫头猫const remove = async (musicItem: IMusic.IMusicItem) => {
3245500cea7S猫头猫    const playList = getPlayList();
3255500cea7S猫头猫    let newPlayList: IMusic.IMusicItem[] = [];
3265500cea7S猫头猫    let currentMusic: IMusic.IMusicItem | null = currentMusicStore.getValue();
3275500cea7S猫头猫    const targetIndex = getMusicIndex(musicItem);
3285500cea7S猫头猫    let shouldPlayCurrent: boolean | null = null;
3295500cea7S猫头猫    if (targetIndex === -1) {
3305500cea7S猫头猫        // 1. 这种情况应该是出错了
3315500cea7S猫头猫        return;
3325500cea7S猫头猫    }
3335500cea7S猫头猫    // 2. 移除的是当前项
3345500cea7S猫头猫    if (currentIndex === targetIndex) {
3355500cea7S猫头猫        // 2.1 停止播放,移除当前项
3365500cea7S猫头猫        newPlayList = produce(playList, draft => {
3375500cea7S猫头猫            draft.splice(targetIndex, 1);
3385500cea7S猫头猫        });
3395500cea7S猫头猫        // 2.2 设置新的播放列表,并更新当前音乐
3405500cea7S猫头猫        if (newPlayList.length === 0) {
3415500cea7S猫头猫            currentMusic = null;
3425500cea7S猫头猫            shouldPlayCurrent = false;
3435500cea7S猫头猫        } else {
3445500cea7S猫头猫            currentMusic = newPlayList[currentIndex % newPlayList.length];
3455500cea7S猫头猫            try {
3465500cea7S猫头猫                const state = (await ReactNativeTrackPlayer.getPlaybackState())
3475500cea7S猫头猫                    .state;
3485500cea7S猫头猫                if (musicIsPaused(state)) {
3495500cea7S猫头猫                    shouldPlayCurrent = false;
3505500cea7S猫头猫                } else {
3515500cea7S猫头猫                    shouldPlayCurrent = true;
3525500cea7S猫头猫                }
3535500cea7S猫头猫            } catch {
3545500cea7S猫头猫                shouldPlayCurrent = false;
3555500cea7S猫头猫            }
3565500cea7S猫头猫        }
3575500cea7S猫头猫    } else {
3585500cea7S猫头猫        // 3. 删除
3595500cea7S猫头猫        newPlayList = produce(playList, draft => {
3605500cea7S猫头猫            draft.splice(targetIndex, 1);
3615500cea7S猫头猫        });
3625500cea7S猫头猫    }
3635500cea7S猫头猫
3645500cea7S猫头猫    setPlayList(newPlayList);
3655500cea7S猫头猫    setCurrentMusic(currentMusic);
3665500cea7S猫头猫    if (shouldPlayCurrent === true) {
3675500cea7S猫头猫        await play(currentMusic, true);
3685500cea7S猫头猫    } else if (shouldPlayCurrent === false) {
3695500cea7S猫头猫        await ReactNativeTrackPlayer.reset();
3705500cea7S猫头猫    }
3715500cea7S猫头猫};
3725500cea7S猫头猫
3735500cea7S猫头猫/**
3745500cea7S猫头猫 * 设置播放模式
3755500cea7S猫头猫 * @param mode 播放模式
3765500cea7S猫头猫 */
3775500cea7S猫头猫const setRepeatMode = (mode: MusicRepeatMode) => {
3785500cea7S猫头猫    const playList = getPlayList();
3795500cea7S猫头猫    let newPlayList;
3805500cea7S猫头猫    if (mode === MusicRepeatMode.SHUFFLE) {
3815500cea7S猫头猫        newPlayList = shuffle(playList);
3825500cea7S猫头猫    } else {
3835500cea7S猫头猫        newPlayList = produce(playList, draft => {
3845500cea7S猫头猫            return sortByTimestampAndIndex(draft);
3855500cea7S猫头猫        });
3865500cea7S猫头猫    }
3875500cea7S猫头猫
3885500cea7S猫头猫    setPlayList(newPlayList);
3895500cea7S猫头猫    const currentMusicItem = currentMusicStore.getValue();
3905500cea7S猫头猫    currentIndex = getMusicIndex(currentMusicItem);
3915500cea7S猫头猫    repeatModeStore.setValue(mode);
3925500cea7S猫头猫    // 更新下一首歌的信息
3935500cea7S猫头猫    ReactNativeTrackPlayer.updateMetadataForTrack(1, getFakeNextTrack());
3945500cea7S猫头猫    // 记录
395*6e000b99S猫头猫    PersistStatus.set('music.repeatMode', mode);
3965500cea7S猫头猫};
3975500cea7S猫头猫
3985500cea7S猫头猫/** 清空播放列表 */
3995500cea7S猫头猫const clear = async () => {
4005500cea7S猫头猫    setPlayList([]);
4015500cea7S猫头猫    setCurrentMusic(null);
4025500cea7S猫头猫
4035500cea7S猫头猫    await ReactNativeTrackPlayer.reset();
404*6e000b99S猫头猫    PersistStatus.set('music.musicItem', undefined);
405*6e000b99S猫头猫    PersistStatus.set('music.progress', 0);
4065500cea7S猫头猫};
4075500cea7S猫头猫
4085500cea7S猫头猫/** 暂停 */
4095500cea7S猫头猫const pause = async () => {
4105500cea7S猫头猫    await ReactNativeTrackPlayer.pause();
4115500cea7S猫头猫};
4125500cea7S猫头猫
4135500cea7S猫头猫const setCurrentMusic = (musicItem?: IMusic.IMusicItem | null) => {
4145500cea7S猫头猫    if (!musicItem) {
4155500cea7S猫头猫        currentIndex = -1;
416f511aee9S猫头猫        currentMusicStore.setValue(null);
417*6e000b99S猫头猫        PersistStatus.set('music.musicItem', undefined);
418*6e000b99S猫头猫        PersistStatus.set('music.progress', 0);
419*6e000b99S猫头猫        return;
4205500cea7S猫头猫    }
4215500cea7S猫头猫    currentIndex = getMusicIndex(musicItem);
422*6e000b99S猫头猫    currentMusicStore.setValue(musicItem);
423*6e000b99S猫头猫
424*6e000b99S猫头猫    PersistStatus.set('music.musicItem', musicItem);
425*6e000b99S猫头猫    PersistStatus.set('music.progress', 0);
4265500cea7S猫头猫};
4275500cea7S猫头猫
428*6e000b99S猫头猫const setQuality = (quality: IMusic.IQualityKey) => {
429*6e000b99S猫头猫    qualityStore.setValue(quality);
430*6e000b99S猫头猫    PersistStatus.set('music.quality', quality);
431*6e000b99S猫头猫};
4325500cea7S猫头猫/**
4335500cea7S猫头猫 * 播放
4345500cea7S猫头猫 *
4355500cea7S猫头猫 * 当musicItem 为空时,代表暂停/播放
4365500cea7S猫头猫 *
4375500cea7S猫头猫 * @param musicItem
4385500cea7S猫头猫 * @param forcePlay
4395500cea7S猫头猫 * @returns
4405500cea7S猫头猫 */
4415500cea7S猫头猫const play = async (
4425500cea7S猫头猫    musicItem?: IMusic.IMusicItem | null,
4435500cea7S猫头猫    forcePlay?: boolean,
4445500cea7S猫头猫) => {
4455500cea7S猫头猫    try {
4465500cea7S猫头猫        if (!musicItem) {
4475500cea7S猫头猫            musicItem = currentMusicStore.getValue();
4485500cea7S猫头猫        }
4495500cea7S猫头猫        if (!musicItem) {
4505500cea7S猫头猫            throw new Error(PlayFailReason.PLAY_LIST_IS_EMPTY);
4515500cea7S猫头猫        }
4525500cea7S猫头猫        // 1. 移动网络禁止播放
4535500cea7S猫头猫        if (
4545500cea7S猫头猫            Network.isCellular() &&
4555500cea7S猫头猫            !Config.get('setting.basic.useCelluarNetworkPlay') &&
4565500cea7S猫头猫            !LocalMusicSheet.isLocalMusic(musicItem)
4575500cea7S猫头猫        ) {
4585500cea7S猫头猫            await ReactNativeTrackPlayer.reset();
4595500cea7S猫头猫            throw new Error(PlayFailReason.FORBID_CELLUAR_NETWORK_PLAY);
4605500cea7S猫头猫        }
4615500cea7S猫头猫
4625500cea7S猫头猫        // 2. 如果是当前正在播放的音频
4635500cea7S猫头猫        if (isCurrentMusic(musicItem)) {
4645500cea7S猫头猫            const currentTrack = await ReactNativeTrackPlayer.getTrack(0);
4655500cea7S猫头猫            // 2.1 如果当前有源
4665500cea7S猫头猫            if (
4675500cea7S猫头猫                currentTrack?.url &&
4685500cea7S猫头猫                isSameMediaItem(musicItem, currentTrack as IMusic.IMusicItem)
4695500cea7S猫头猫            ) {
4705500cea7S猫头猫                const currentActiveIndex =
4715500cea7S猫头猫                    await ReactNativeTrackPlayer.getActiveTrackIndex();
4725500cea7S猫头猫                if (currentActiveIndex !== 0) {
4735500cea7S猫头猫                    await ReactNativeTrackPlayer.skip(0);
4745500cea7S猫头猫                }
4755500cea7S猫头猫                if (forcePlay) {
4765500cea7S猫头猫                    // 2.1.1 强制重新开始
4775500cea7S猫头猫                    await ReactNativeTrackPlayer.seekTo(0);
4786f57784cS猫头猫                }
4796f57784cS猫头猫                if (
4805500cea7S猫头猫                    (await ReactNativeTrackPlayer.getPlaybackState()).state !==
4815500cea7S猫头猫                    State.Playing
4825500cea7S猫头猫                ) {
4835500cea7S猫头猫                    // 2.1.2 恢复播放
4845500cea7S猫头猫                    await ReactNativeTrackPlayer.play();
4855500cea7S猫头猫                }
4865500cea7S猫头猫                // 这种情况下,播放队列和当前歌曲都不需要变化
4875500cea7S猫头猫                return;
4885500cea7S猫头猫            }
4895500cea7S猫头猫            // 2.2 其他情况:重新获取源
4905500cea7S猫头猫        }
4915500cea7S猫头猫
4925500cea7S猫头猫        // 3. 如果没有在播放列表中,添加到队尾;同时更新列表状态
4935500cea7S猫头猫        const inPlayList = isInPlayList(musicItem);
4945500cea7S猫头猫        if (!inPlayList) {
4955500cea7S猫头猫            add(musicItem);
4965500cea7S猫头猫        }
4975500cea7S猫头猫
4985500cea7S猫头猫        // 4. 更新列表状态和当前音乐
4995500cea7S猫头猫        setCurrentMusic(musicItem);
5005500cea7S猫头猫
5015500cea7S猫头猫        // 5. 获取音源
5025500cea7S猫头猫        let track: IMusic.IMusicItem;
5035500cea7S猫头猫
5045500cea7S猫头猫        // 5.1 通过插件获取音源
5055500cea7S猫头猫        const plugin = PluginManager.getByName(musicItem.platform);
5065500cea7S猫头猫        // 5.2 获取音质排序
5075500cea7S猫头猫        const qualityOrder = getQualityOrder(
5085500cea7S猫头猫            Config.get('setting.basic.defaultPlayQuality') ?? 'standard',
5095500cea7S猫头猫            Config.get('setting.basic.playQualityOrder') ?? 'asc',
5105500cea7S猫头猫        );
5115500cea7S猫头猫        // 5.3 插件返回音源
5125500cea7S猫头猫        let source: IPlugin.IMediaSourceResult | null = null;
5135500cea7S猫头猫        for (let quality of qualityOrder) {
5145500cea7S猫头猫            if (isCurrentMusic(musicItem)) {
5155500cea7S猫头猫                source =
5165500cea7S猫头猫                    (await plugin?.methods?.getMediaSource(
5175500cea7S猫头猫                        musicItem,
5185500cea7S猫头猫                        quality,
5195500cea7S猫头猫                    )) ?? null;
5205500cea7S猫头猫                // 5.3.1 获取到真实源
5215500cea7S猫头猫                if (source) {
522*6e000b99S猫头猫                    setQuality(quality);
523*6e000b99S猫头猫
5245500cea7S猫头猫                    break;
5255500cea7S猫头猫                }
5265500cea7S猫头猫            } else {
5275500cea7S猫头猫                // 5.3.2 已经切换到其他歌曲了,
5285500cea7S猫头猫                return;
5295500cea7S猫头猫            }
5305500cea7S猫头猫        }
5315500cea7S猫头猫
5325500cea7S猫头猫        if (!isCurrentMusic(musicItem)) {
5335500cea7S猫头猫            return;
5345500cea7S猫头猫        }
5355500cea7S猫头猫
5365500cea7S猫头猫        if (!source) {
53743eb30bfS猫头猫            // 如果有source
53843eb30bfS猫头猫            if (musicItem.source) {
53943eb30bfS猫头猫                for (let quality of qualityOrder) {
54043eb30bfS猫头猫                    if (musicItem.source[quality]?.url) {
54143eb30bfS猫头猫                        source = musicItem.source[quality]!;
542*6e000b99S猫头猫                        setQuality(quality);
543*6e000b99S猫头猫
54443eb30bfS猫头猫                        break;
5455500cea7S猫头猫                    }
54643eb30bfS猫头猫                }
54743eb30bfS猫头猫            }
54843eb30bfS猫头猫
54943eb30bfS猫头猫            // 5.4 没有返回源
55043eb30bfS猫头猫            if (!source && !musicItem.url) {
55143eb30bfS猫头猫                throw new Error(PlayFailReason.INVALID_SOURCE);
55243eb30bfS猫头猫            } else {
5535500cea7S猫头猫                source = {
5545500cea7S猫头猫                    url: musicItem.url,
5555500cea7S猫头猫                };
556*6e000b99S猫头猫                setQuality('standard');
5575500cea7S猫头猫            }
55843eb30bfS猫头猫        }
5595500cea7S猫头猫
5605500cea7S猫头猫        // 6. 特殊类型源
5615500cea7S猫头猫        if (getUrlExt(source.url) === '.m3u8') {
5625500cea7S猫头猫            // @ts-ignore
5635500cea7S猫头猫            source.type = 'hls';
5645500cea7S猫头猫        }
5655500cea7S猫头猫        // 7. 合并结果
5665500cea7S猫头猫        track = mergeProps(musicItem, source) as IMusic.IMusicItem;
5675500cea7S猫头猫
5685500cea7S猫头猫        // 8. 新增历史记录
5695500cea7S猫头猫        musicHistory.addMusic(musicItem);
5705500cea7S猫头猫
57162e73a5eS猫头猫        trace('获取音源成功', track);
57262e73a5eS猫头猫
5735500cea7S猫头猫        // 9. 设置音源
5745500cea7S猫头猫        await setTrackSource(track as Track);
5755500cea7S猫头猫
5765500cea7S猫头猫        // 10. 获取补充信息
5775500cea7S猫头猫        let info: Partial<IMusic.IMusicItem> | null = null;
5785500cea7S猫头猫        try {
5795500cea7S猫头猫            info = (await plugin?.methods?.getMusicInfo?.(musicItem)) ?? null;
5805500cea7S猫头猫        } catch {}
5815500cea7S猫头猫
5825500cea7S猫头猫        // 11. 设置补充信息
5835500cea7S猫头猫        if (info && isCurrentMusic(musicItem)) {
5845500cea7S猫头猫            const mergedTrack = mergeProps(track, info);
5855500cea7S猫头猫            currentMusicStore.setValue(mergedTrack as IMusic.IMusicItem);
5865500cea7S猫头猫            await ReactNativeTrackPlayer.updateMetadataForTrack(
5875500cea7S猫头猫                0,
5885500cea7S猫头猫                mergedTrack as TrackMetadataBase,
5895500cea7S猫头猫            );
5905500cea7S猫头猫        }
5915500cea7S猫头猫
5925500cea7S猫头猫        // 12. 刷新歌词信息
5935500cea7S猫头猫        if (
5945500cea7S猫头猫            !isSameMediaItem(
5955500cea7S猫头猫                LyricManager.getLyricState()?.lyricParser?.getCurrentMusicItem?.(),
5965500cea7S猫头猫                musicItem,
5975500cea7S猫头猫            )
5985500cea7S猫头猫        ) {
5995500cea7S猫头猫            DeviceEventEmitter.emit(EDeviceEvents.REFRESH_LYRIC, true);
6005500cea7S猫头猫        }
6015500cea7S猫头猫    } catch (e: any) {
6025500cea7S猫头猫        const message = e?.message;
6035500cea7S猫头猫        if (
6045500cea7S猫头猫            message === 'The player is not initialized. Call setupPlayer first.'
6055500cea7S猫头猫        ) {
6065500cea7S猫头猫            await ReactNativeTrackPlayer.setupPlayer();
6075500cea7S猫头猫            play(musicItem, forcePlay);
6085500cea7S猫头猫        } else if (message === PlayFailReason.FORBID_CELLUAR_NETWORK_PLAY) {
609f511aee9S猫头猫            Toast.warn(
610f511aee9S猫头猫                '当前禁止移动网络播放音乐,如需播放请去侧边栏-基本设置中修改',
611f511aee9S猫头猫            );
6125500cea7S猫头猫        } else if (message === PlayFailReason.INVALID_SOURCE) {
61362e73a5eS猫头猫            trace('音源为空,播放失败');
6146f57784cS猫头猫            await failToPlay();
6155500cea7S猫头猫        } else if (message === PlayFailReason.PLAY_LIST_IS_EMPTY) {
6165500cea7S猫头猫            // 队列是空的,不应该出现这种情况
6175500cea7S猫头猫        }
6185500cea7S猫头猫    }
6195500cea7S猫头猫};
6205500cea7S猫头猫
6215500cea7S猫头猫/**
6225500cea7S猫头猫 * 播放音乐,同时替换播放队列
6235500cea7S猫头猫 * @param musicItem 音乐
6245500cea7S猫头猫 * @param newPlayList 替代列表
6255500cea7S猫头猫 */
6265500cea7S猫头猫const playWithReplacePlayList = async (
6275500cea7S猫头猫    musicItem: IMusic.IMusicItem,
6285500cea7S猫头猫    newPlayList: IMusic.IMusicItem[],
6295500cea7S猫头猫) => {
6305500cea7S猫头猫    if (newPlayList.length !== 0) {
6315500cea7S猫头猫        const now = Date.now();
6325500cea7S猫头猫        if (newPlayList.length > maxMusicQueueLength) {
6335500cea7S猫头猫            newPlayList = shrinkPlayListToSize(
6345500cea7S猫头猫                newPlayList,
6355500cea7S猫头猫                newPlayList.findIndex(it => isSameMediaItem(it, musicItem)),
6365500cea7S猫头猫            );
6375500cea7S猫头猫        }
6385500cea7S猫头猫        const playListItems = newPlayList.map((item, index) =>
6395500cea7S猫头猫            produce(item, draft => {
6405500cea7S猫头猫                draft[timeStampSymbol] = now;
6415500cea7S猫头猫                draft[sortIndexSymbol] = index;
6425500cea7S猫头猫            }),
6435500cea7S猫头猫        );
6445500cea7S猫头猫        setPlayList(
6455500cea7S猫头猫            repeatModeStore.getValue() === MusicRepeatMode.SHUFFLE
6465500cea7S猫头猫                ? shuffle(playListItems)
6475500cea7S猫头猫                : playListItems,
6485500cea7S猫头猫        );
6495500cea7S猫头猫        await play(musicItem, true);
6505500cea7S猫头猫    }
6515500cea7S猫头猫};
6525500cea7S猫头猫
6536f57784cS猫头猫const skipToNext = async () => {
6545500cea7S猫头猫    if (isPlayListEmpty()) {
6555500cea7S猫头猫        setCurrentMusic(null);
6565500cea7S猫头猫        return;
6575500cea7S猫头猫    }
6585500cea7S猫头猫
6595500cea7S猫头猫    await play(getPlayListMusicAt(currentIndex + 1), true);
6605500cea7S猫头猫};
6615500cea7S猫头猫
6625500cea7S猫头猫const skipToPrevious = async () => {
6635500cea7S猫头猫    if (isPlayListEmpty()) {
6645500cea7S猫头猫        setCurrentMusic(null);
6655500cea7S猫头猫        return;
6665500cea7S猫头猫    }
6675500cea7S猫头猫
6686f57784cS猫头猫    await play(
6696f57784cS猫头猫        getPlayListMusicAt(currentIndex === -1 ? 0 : currentIndex - 1),
6706f57784cS猫头猫        true,
6716f57784cS猫头猫    );
6725500cea7S猫头猫};
6735500cea7S猫头猫
6745500cea7S猫头猫/** 修改当前播放的音质 */
6755500cea7S猫头猫const changeQuality = async (newQuality: IMusic.IQualityKey) => {
6765500cea7S猫头猫    // 获取当前的音乐和进度
6775500cea7S猫头猫    if (newQuality === qualityStore.getValue()) {
6785500cea7S猫头猫        return true;
6795500cea7S猫头猫    }
6805500cea7S猫头猫
6815500cea7S猫头猫    // 获取当前歌曲
6825500cea7S猫头猫    const musicItem = currentMusicStore.getValue();
6835500cea7S猫头猫    if (!musicItem) {
6845500cea7S猫头猫        return false;
6855500cea7S猫头猫    }
6865500cea7S猫头猫    try {
6875500cea7S猫头猫        const progress = await ReactNativeTrackPlayer.getProgress();
6885500cea7S猫头猫        const plugin = PluginManager.getByMedia(musicItem);
6895500cea7S猫头猫        const newSource = await plugin?.methods?.getMediaSource(
6905500cea7S猫头猫            musicItem,
6915500cea7S猫头猫            newQuality,
6925500cea7S猫头猫        );
6935500cea7S猫头猫        if (!newSource?.url) {
6945500cea7S猫头猫            throw new Error(PlayFailReason.INVALID_SOURCE);
6955500cea7S猫头猫        }
6965500cea7S猫头猫        if (isCurrentMusic(musicItem)) {
6975500cea7S猫头猫            const playingState = (
6985500cea7S猫头猫                await ReactNativeTrackPlayer.getPlaybackState()
6995500cea7S猫头猫            ).state;
7005500cea7S猫头猫            await setTrackSource(
7015500cea7S猫头猫                mergeProps(musicItem, newSource) as unknown as Track,
7025500cea7S猫头猫                !musicIsPaused(playingState),
7035500cea7S猫头猫            );
7045500cea7S猫头猫
7055500cea7S猫头猫            await ReactNativeTrackPlayer.seekTo(progress.position ?? 0);
706*6e000b99S猫头猫            setQuality(newQuality);
7075500cea7S猫头猫        }
7085500cea7S猫头猫        return true;
7095500cea7S猫头猫    } catch {
7105500cea7S猫头猫        // 修改失败
7115500cea7S猫头猫        return false;
7125500cea7S猫头猫    }
7135500cea7S猫头猫};
7145500cea7S猫头猫
7155500cea7S猫头猫enum PlayFailReason {
7165500cea7S猫头猫    /** 禁止移动网络播放 */
7175500cea7S猫头猫    FORBID_CELLUAR_NETWORK_PLAY = 'FORBID_CELLUAR_NETWORK_PLAY',
7185500cea7S猫头猫    /** 播放列表为空 */
7195500cea7S猫头猫    PLAY_LIST_IS_EMPTY = 'PLAY_LIST_IS_EMPTY',
7205500cea7S猫头猫    /** 无效源 */
7215500cea7S猫头猫    INVALID_SOURCE = 'INVALID_SOURCE',
7225500cea7S猫头猫    /** 非当前音乐 */
7235500cea7S猫头猫}
7245500cea7S猫头猫
7255500cea7S猫头猫function useMusicState() {
7265500cea7S猫头猫    const playbackState = usePlaybackState();
7275500cea7S猫头猫
7285500cea7S猫头猫    return playbackState.state;
7295500cea7S猫头猫}
7305500cea7S猫头猫
731f511aee9S猫头猫function getPreviousMusic() {
732f511aee9S猫头猫    const currentMusicItem = currentMusicStore.getValue();
733f511aee9S猫头猫    if (!currentMusicItem) {
734f511aee9S猫头猫        return null;
735f511aee9S猫头猫    }
736f511aee9S猫头猫
737f511aee9S猫头猫    return getPlayListMusicAt(currentIndex - 1);
738f511aee9S猫头猫}
739f511aee9S猫头猫
740f511aee9S猫头猫function getNextMusic() {
741f511aee9S猫头猫    const currentMusicItem = currentMusicStore.getValue();
742f511aee9S猫头猫    if (!currentMusicItem) {
743f511aee9S猫头猫        return null;
744f511aee9S猫头猫    }
745f511aee9S猫头猫
746f511aee9S猫头猫    return getPlayListMusicAt(currentIndex + 1);
747f511aee9S猫头猫}
748f511aee9S猫头猫
7495500cea7S猫头猫const TrackPlayer = {
7505500cea7S猫头猫    setupTrackPlayer,
7515500cea7S猫头猫    usePlayList,
7525500cea7S猫头猫    getPlayList,
7535500cea7S猫头猫    addAll,
7545500cea7S猫头猫    add,
7555500cea7S猫头猫    addNext,
7565500cea7S猫头猫    skipToNext,
7575500cea7S猫头猫    skipToPrevious,
7585500cea7S猫头猫    play,
7595500cea7S猫头猫    playWithReplacePlayList,
7605500cea7S猫头猫    pause,
7615500cea7S猫头猫    remove,
7625500cea7S猫头猫    clear,
7635500cea7S猫头猫    useCurrentMusic: currentMusicStore.useValue,
7645500cea7S猫头猫    getCurrentMusic: currentMusicStore.getValue,
7655500cea7S猫头猫    useRepeatMode: repeatModeStore.useValue,
7665500cea7S猫头猫    getRepeatMode: repeatModeStore.getValue,
7675500cea7S猫头猫    toggleRepeatMode,
7685500cea7S猫头猫    usePlaybackState,
7695500cea7S猫头猫    getProgress: ReactNativeTrackPlayer.getProgress,
7705500cea7S猫头猫    useProgress: useProgress,
7715500cea7S猫头猫    seekTo: ReactNativeTrackPlayer.seekTo,
7725500cea7S猫头猫    changeQuality,
7735500cea7S猫头猫    useCurrentQuality: qualityStore.useValue,
7745500cea7S猫头猫    getCurrentQuality: qualityStore.getValue,
7755500cea7S猫头猫    getRate: ReactNativeTrackPlayer.getRate,
7765500cea7S猫头猫    setRate: ReactNativeTrackPlayer.setRate,
7775500cea7S猫头猫    useMusicState,
7785500cea7S猫头猫    reset: ReactNativeTrackPlayer.reset,
779f511aee9S猫头猫    getPreviousMusic,
780f511aee9S猫头猫    getNextMusic,
7815500cea7S猫头猫};
7825500cea7S猫头猫
7835500cea7S猫头猫export default TrackPlayer;
7845500cea7S猫头猫export {MusicRepeatMode, State as MusicState};
785