xref: /MusicFree/src/core/trackPlayer/index.ts (revision 41ddce918e1138d8f16e522cc7c19ac86ceca698)
163c74e8eSmaotoumaoimport { produce } from "immer";
25500cea7S猫头猫import ReactNativeTrackPlayer, {
35500cea7S猫头猫    Event,
45500cea7S猫头猫    State,
55500cea7S猫头猫    Track,
65500cea7S猫头猫    TrackMetadataBase,
75500cea7S猫头猫    usePlaybackState,
863c74e8eSmaotoumao    useProgress
963c74e8eSmaotoumao} from "react-native-track-player";
1063c74e8eSmaotoumaoimport shuffle from "lodash.shuffle";
11*41ddce91Smaotoumaoimport Config from "../config.ts";
1263c74e8eSmaotoumaoimport { EDeviceEvents, internalFakeSoundKey, sortIndexSymbol, timeStampSymbol } from "@/constants/commonConst";
1363c74e8eSmaotoumaoimport { GlobalState } from "@/utils/stateMapper";
1463c74e8eSmaotoumaoimport delay from "@/utils/delay";
1563c74e8eSmaotoumaoimport { isSameMediaItem, mergeProps, sortByTimestampAndIndex } from "@/utils/mediaItem";
1663c74e8eSmaotoumaoimport Network from "../network";
1763c74e8eSmaotoumaoimport LocalMusicSheet from "../localMusicSheet";
1863c74e8eSmaotoumaoimport { SoundAsset } from "@/constants/assetsConst";
1963c74e8eSmaotoumaoimport { getQualityOrder } from "@/utils/qualities";
2063c74e8eSmaotoumaoimport musicHistory from "../musicHistory";
2163c74e8eSmaotoumaoimport getUrlExt from "@/utils/getUrlExt";
2263c74e8eSmaotoumaoimport { DeviceEventEmitter } from "react-native";
2363c74e8eSmaotoumaoimport LyricManager from "../lyricManager";
2463c74e8eSmaotoumaoimport { MusicRepeatMode } from "./common";
255500cea7S猫头猫import {
265500cea7S猫头猫    getMusicIndex,
275500cea7S猫头猫    getPlayList,
285500cea7S猫头猫    getPlayListMusicAt,
295500cea7S猫头猫    isInPlayList,
305500cea7S猫头猫    isPlayListEmpty,
315500cea7S猫头猫    setPlayList,
3263c74e8eSmaotoumao    usePlayList
3363c74e8eSmaotoumao} from "./internal/playList";
3463c74e8eSmaotoumaoimport { createMediaIndexMap } from "@/utils/mediaIndexMap";
3563c74e8eSmaotoumaoimport PluginManager from "../pluginManager";
3663c74e8eSmaotoumaoimport { musicIsPaused } from "@/utils/trackUtils";
3763c74e8eSmaotoumaoimport { errorLog, trace } from "@/utils/log";
38819a9075Smaotoumaoimport PersistStatus from "../persistStatus.ts";
3963c74e8eSmaotoumaoimport { getCurrentDialog, showDialog } from "@/components/dialogs/useDialog";
4063c74e8eSmaotoumaoimport getSimilarMusic from "@/utils/getSimilarMusic";
415500cea7S猫头猫
425500cea7S猫头猫/** 当前播放 */
435500cea7S猫头猫const currentMusicStore = new GlobalState<IMusic.IMusicItem | null>(null);
445500cea7S猫头猫
455500cea7S猫头猫/** 播放模式 */
465500cea7S猫头猫const repeatModeStore = new GlobalState<MusicRepeatMode>(MusicRepeatMode.QUEUE);
475500cea7S猫头猫
485500cea7S猫头猫/** 音质 */
495500cea7S猫头猫const qualityStore = new GlobalState<IMusic.IQualityKey>('standard');
505500cea7S猫头猫
515500cea7S猫头猫let currentIndex = -1;
525500cea7S猫头猫
534c222b69S猫头猫const maxMusicQueueLength = 10000; // 当前播放最大限制
545500cea7S猫头猫const halfMaxMusicQueueLength = Math.floor(maxMusicQueueLength / 2);
555500cea7S猫头猫const shrinkPlayListToSize = (
565500cea7S猫头猫    queue: IMusic.IMusicItem[],
575500cea7S猫头猫    targetIndex = currentIndex,
585500cea7S猫头猫) => {
595500cea7S猫头猫    // 播放列表上限,太多无法缓存状态
605500cea7S猫头猫    if (queue.length > maxMusicQueueLength) {
615500cea7S猫头猫        if (targetIndex < halfMaxMusicQueueLength) {
625500cea7S猫头猫            queue = queue.slice(0, maxMusicQueueLength);
635500cea7S猫头猫        } else {
645500cea7S猫头猫            const right = Math.min(
655500cea7S猫头猫                queue.length,
665500cea7S猫头猫                targetIndex + halfMaxMusicQueueLength,
675500cea7S猫头猫            );
685500cea7S猫头猫            const left = Math.max(0, right - maxMusicQueueLength);
695500cea7S猫头猫            queue = queue.slice(left, right);
705500cea7S猫头猫        }
715500cea7S猫头猫    }
725500cea7S猫头猫    return queue;
735500cea7S猫头猫};
745500cea7S猫头猫
757aed04d4S猫头猫let hasSetupListener = false;
767aed04d4S猫头猫
776e000b99S猫头猫async function setupTrackPlayer() {
78819a9075Smaotoumao    const rate = PersistStatus.get('music.rate');
79819a9075Smaotoumao    const musicQueue = PersistStatus.get('music.playList');
80819a9075Smaotoumao    const repeatMode = PersistStatus.get('music.repeatMode');
81819a9075Smaotoumao    const progress = PersistStatus.get('music.progress');
82819a9075Smaotoumao    const track = PersistStatus.get('music.musicItem');
836e000b99S猫头猫    const quality =
84819a9075Smaotoumao        PersistStatus.get('music.quality') ||
85*41ddce91Smaotoumao        Config.getConfig('basic.defaultPlayQuality') ||
866e000b99S猫头猫        'standard';
875500cea7S猫头猫
885500cea7S猫头猫    // 状态恢复
895500cea7S猫头猫    if (rate) {
9091eb8fa8S猫头猫        ReactNativeTrackPlayer.setRate(+rate / 100);
915500cea7S猫头猫    }
92e476f1a0S猫头猫    if (repeatMode) {
93e476f1a0S猫头猫        repeatModeStore.setValue(repeatMode as MusicRepeatMode);
94e476f1a0S猫头猫    }
955500cea7S猫头猫
965500cea7S猫头猫    if (musicQueue && Array.isArray(musicQueue)) {
975500cea7S猫头猫        addAll(musicQueue, undefined, repeatMode === MusicRepeatMode.SHUFFLE);
985500cea7S猫头猫    }
995500cea7S猫头猫
1005500cea7S猫头猫    if (track && isInPlayList(track)) {
101*41ddce91Smaotoumao        if (!Config.getConfig('basic.autoPlayWhenAppStart')) {
1028b9904b3S猫头猫            track.isInit = true;
1038b9904b3S猫头猫        }
1040dd58d38S猫头猫
10591eb8fa8S猫头猫        // 异步
10691eb8fa8S猫头猫        PluginManager.getByMedia(track)
10791eb8fa8S猫头猫            ?.methods.getMediaSource(track, quality, 0)
10891eb8fa8S猫头猫            .then(async newSource => {
10991eb8fa8S猫头猫                track.url = newSource?.url || track.url;
11091eb8fa8S猫头猫                track.headers = newSource?.headers || track.headers;
11191eb8fa8S猫头猫
11291eb8fa8S猫头猫                if (isSameMediaItem(currentMusicStore.getValue(), track)) {
1135500cea7S猫头猫                    await setTrackSource(track as Track, false);
11491eb8fa8S猫头猫                }
11591eb8fa8S猫头猫            });
1165500cea7S猫头猫        setCurrentMusic(track);
1175500cea7S猫头猫
1186e000b99S猫头猫        if (progress) {
11991eb8fa8S猫头猫            // 异步
12091eb8fa8S猫头猫            ReactNativeTrackPlayer.seekTo(progress);
1215500cea7S猫头猫        }
1225500cea7S猫头猫    }
1235500cea7S猫头猫
1247aed04d4S猫头猫    if (!hasSetupListener) {
1255500cea7S猫头猫        ReactNativeTrackPlayer.addEventListener(
1265500cea7S猫头猫            Event.PlaybackActiveTrackChanged,
1275500cea7S猫头猫            async evt => {
1285500cea7S猫头猫                if (
1295500cea7S猫头猫                    evt.index === 1 &&
1305500cea7S猫头猫                    evt.lastIndex === 0 &&
1315500cea7S猫头猫                    evt.track?.$ === internalFakeSoundKey
1325500cea7S猫头猫                ) {
13362e73a5eS猫头猫                    trace('队列末尾,播放下一首');
1345500cea7S猫头猫                    if (repeatModeStore.getValue() === MusicRepeatMode.SINGLE) {
1355500cea7S猫头猫                        await play(null, true);
1365500cea7S猫头猫                    } else {
1375500cea7S猫头猫                        // 当前生效的歌曲是下一曲的标记
1386f57784cS猫头猫                        await skipToNext();
1395500cea7S猫头猫                    }
1405500cea7S猫头猫                }
1415500cea7S猫头猫            },
1425500cea7S猫头猫        );
1435500cea7S猫头猫
1447aed04d4S猫头猫        ReactNativeTrackPlayer.addEventListener(
1457aed04d4S猫头猫            Event.PlaybackError,
14662e73a5eS猫头猫            async e => {
14717958235S猫头猫                errorLog('播放出错', e.message);
14862e73a5eS猫头猫                // WARNING: 不稳定,报错的时候有可能track已经变到下一首歌去了
1498b9904b3S猫头猫                const currentTrack =
1508b9904b3S猫头猫                    await ReactNativeTrackPlayer.getActiveTrack();
1518b9904b3S猫头猫                if (currentTrack?.isInit) {
1528b9904b3S猫头猫                    // HACK: 避免初始失败的情况
1538b9904b3S猫头猫                    ReactNativeTrackPlayer.updateMetadataForTrack(0, {
1548b9904b3S猫头猫                        ...currentTrack,
1558b9904b3S猫头猫                        // @ts-ignore
1568b9904b3S猫头猫                        isInit: undefined,
1578b9904b3S猫头猫                    });
1588b9904b3S猫头猫                    return;
1598b9904b3S猫头猫                }
1608b9904b3S猫头猫
1617aed04d4S猫头猫                if (
16262e73a5eS猫头猫                    (await ReactNativeTrackPlayer.getActiveTrackIndex()) ===
16362e73a5eS猫头猫                        0 &&
16462e73a5eS猫头猫                    e.message &&
16562e73a5eS猫头猫                    e.message !== 'android-io-file-not-found'
1667aed04d4S猫头猫                ) {
16762e73a5eS猫头猫                    trace('播放出错', {
16862e73a5eS猫头猫                        message: e.message,
16962e73a5eS猫头猫                        code: e.code,
17062e73a5eS猫头猫                    });
1718b9904b3S猫头猫
1725500cea7S猫头猫                    failToPlay();
1735500cea7S猫头猫                }
1747aed04d4S猫头猫            },
1757aed04d4S猫头猫        );
1767aed04d4S猫头猫
1777aed04d4S猫头猫        hasSetupListener = true;
1787aed04d4S猫头猫    }
1795500cea7S猫头猫}
1805500cea7S猫头猫
1815500cea7S猫头猫/**
1825500cea7S猫头猫 * 获取自动播放的下一个track
1835500cea7S猫头猫 */
1845500cea7S猫头猫const getFakeNextTrack = () => {
1855500cea7S猫头猫    let track: Track | undefined;
1865500cea7S猫头猫    const repeatMode = repeatModeStore.getValue();
1875500cea7S猫头猫    if (repeatMode === MusicRepeatMode.SINGLE) {
1885500cea7S猫头猫        // 单曲循环
1895500cea7S猫头猫        track = getPlayListMusicAt(currentIndex) as Track;
1905500cea7S猫头猫    } else {
1915500cea7S猫头猫        // 下一曲
1925500cea7S猫头猫        track = getPlayListMusicAt(currentIndex + 1) as Track;
1935500cea7S猫头猫    }
1945500cea7S猫头猫
1955500cea7S猫头猫    if (track) {
1965500cea7S猫头猫        return produce(track, _ => {
1975500cea7S猫头猫            _.url = SoundAsset.fakeAudio;
1985500cea7S猫头猫            _.$ = internalFakeSoundKey;
199cc77e86bS猫头猫            if (!_.artwork?.trim()?.length) {
200cc77e86bS猫头猫                _.artwork = undefined;
201cc77e86bS猫头猫            }
2025500cea7S猫头猫        });
2035500cea7S猫头猫    } else {
2045500cea7S猫头猫        // 只有列表长度为0时才会出现的特殊情况
2055500cea7S猫头猫        return {url: SoundAsset.fakeAudio, $: internalFakeSoundKey} as Track;
2065500cea7S猫头猫    }
2075500cea7S猫头猫};
2085500cea7S猫头猫
2095500cea7S猫头猫/** 播放失败时的情况 */
2106f57784cS猫头猫async function failToPlay() {
2115500cea7S猫头猫    // 如果自动跳转下一曲, 500s后自动跳转
212*41ddce91Smaotoumao    if (!Config.getConfig('basic.autoStopWhenError')) {
2135500cea7S猫头猫        await ReactNativeTrackPlayer.reset();
2145500cea7S猫头猫        await delay(500);
2156f57784cS猫头猫        await skipToNext();
2165500cea7S猫头猫    }
2175500cea7S猫头猫}
2185500cea7S猫头猫
2195500cea7S猫头猫// 播放模式相关
2205500cea7S猫头猫const _toggleRepeatMapping = {
2215500cea7S猫头猫    [MusicRepeatMode.SHUFFLE]: MusicRepeatMode.SINGLE,
2225500cea7S猫头猫    [MusicRepeatMode.SINGLE]: MusicRepeatMode.QUEUE,
2235500cea7S猫头猫    [MusicRepeatMode.QUEUE]: MusicRepeatMode.SHUFFLE,
2245500cea7S猫头猫};
2255500cea7S猫头猫/** 切换下一个模式 */
2265500cea7S猫头猫const toggleRepeatMode = () => {
2275500cea7S猫头猫    setRepeatMode(_toggleRepeatMapping[repeatModeStore.getValue()]);
2285500cea7S猫头猫};
2295500cea7S猫头猫
2305500cea7S猫头猫/**
2315500cea7S猫头猫 * 添加到播放列表
2325500cea7S猫头猫 * @param musicItems 目标歌曲
2335500cea7S猫头猫 * @param beforeIndex 在第x首歌曲前添加
2345500cea7S猫头猫 * @param shouldShuffle 随机排序
2355500cea7S猫头猫 */
2365500cea7S猫头猫const addAll = (
2375500cea7S猫头猫    musicItems: Array<IMusic.IMusicItem> = [],
2385500cea7S猫头猫    beforeIndex?: number,
2395500cea7S猫头猫    shouldShuffle?: boolean,
2405500cea7S猫头猫) => {
2415500cea7S猫头猫    const now = Date.now();
2425500cea7S猫头猫    let newPlayList: IMusic.IMusicItem[] = [];
2435500cea7S猫头猫    let currentPlayList = getPlayList();
2444c222b69S猫头猫    musicItems.forEach((item, index) => {
2454c222b69S猫头猫        item[timeStampSymbol] = now;
2464c222b69S猫头猫        item[sortIndexSymbol] = index;
2474c222b69S猫头猫    });
2484c222b69S猫头猫
2495500cea7S猫头猫    if (beforeIndex === undefined || beforeIndex < 0) {
2505500cea7S猫头猫        // 1.1. 添加到歌单末尾,并过滤掉已有的歌曲
2515500cea7S猫头猫        newPlayList = currentPlayList.concat(
2524c222b69S猫头猫            musicItems.filter(item => !isInPlayList(item)),
2535500cea7S猫头猫        );
2545500cea7S猫头猫    } else {
2555500cea7S猫头猫        // 1.2. 新的播放列表,插入
2564c222b69S猫头猫        const indexMap = createMediaIndexMap(musicItems);
2575500cea7S猫头猫        const beforeDraft = currentPlayList
2585500cea7S猫头猫            .slice(0, beforeIndex)
2595500cea7S猫头猫            .filter(item => !indexMap.has(item));
2605500cea7S猫头猫        const afterDraft = currentPlayList
2615500cea7S猫头猫            .slice(beforeIndex)
2625500cea7S猫头猫            .filter(item => !indexMap.has(item));
2635500cea7S猫头猫
2644c222b69S猫头猫        newPlayList = [...beforeDraft, ...musicItems, ...afterDraft];
2655500cea7S猫头猫    }
26615900d05S猫头猫
2675500cea7S猫头猫    // 如果太长了
2685500cea7S猫头猫    if (newPlayList.length > maxMusicQueueLength) {
2695500cea7S猫头猫        newPlayList = shrinkPlayListToSize(
2705500cea7S猫头猫            newPlayList,
2715500cea7S猫头猫            beforeIndex ?? newPlayList.length - 1,
2725500cea7S猫头猫        );
2735500cea7S猫头猫    }
2745500cea7S猫头猫
2755500cea7S猫头猫    // 2. 如果需要随机
2765500cea7S猫头猫    if (shouldShuffle) {
2775500cea7S猫头猫        newPlayList = shuffle(newPlayList);
2785500cea7S猫头猫    }
2795500cea7S猫头猫    // 3. 设置播放列表
2805500cea7S猫头猫    setPlayList(newPlayList);
2815500cea7S猫头猫    const currentMusicItem = currentMusicStore.getValue();
2825500cea7S猫头猫
2835500cea7S猫头猫    // 4. 重置下标
2845500cea7S猫头猫    if (currentMusicItem) {
2855500cea7S猫头猫        currentIndex = getMusicIndex(currentMusicItem);
2865500cea7S猫头猫    }
2875500cea7S猫头猫};
2885500cea7S猫头猫
2895500cea7S猫头猫/** 追加到队尾 */
2905500cea7S猫头猫const add = (
2915500cea7S猫头猫    musicItem: IMusic.IMusicItem | IMusic.IMusicItem[],
2925500cea7S猫头猫    beforeIndex?: number,
2935500cea7S猫头猫) => {
2945500cea7S猫头猫    addAll(Array.isArray(musicItem) ? musicItem : [musicItem], beforeIndex);
2955500cea7S猫头猫};
2965500cea7S猫头猫
2975500cea7S猫头猫/**
2985500cea7S猫头猫 * 下一首播放
2995500cea7S猫头猫 * @param musicItem
3005500cea7S猫头猫 */
3015500cea7S猫头猫const addNext = (musicItem: IMusic.IMusicItem | IMusic.IMusicItem[]) => {
3025500cea7S猫头猫    const shouldPlay = isPlayListEmpty();
3035500cea7S猫头猫    add(musicItem, currentIndex + 1);
3045500cea7S猫头猫    if (shouldPlay) {
3055500cea7S猫头猫        play(Array.isArray(musicItem) ? musicItem[0] : musicItem);
3065500cea7S猫头猫    }
3075500cea7S猫头猫};
3085500cea7S猫头猫
3093991724eS猫头猫const isCurrentMusic = (musicItem: IMusic.IMusicItem | null | undefined) => {
3105500cea7S猫头猫    return isSameMediaItem(musicItem, currentMusicStore.getValue()) ?? false;
3115500cea7S猫头猫};
3125500cea7S猫头猫
3135500cea7S猫头猫const remove = async (musicItem: IMusic.IMusicItem) => {
3145500cea7S猫头猫    const playList = getPlayList();
3155500cea7S猫头猫    let newPlayList: IMusic.IMusicItem[] = [];
3165500cea7S猫头猫    let currentMusic: IMusic.IMusicItem | null = currentMusicStore.getValue();
3175500cea7S猫头猫    const targetIndex = getMusicIndex(musicItem);
3185500cea7S猫头猫    let shouldPlayCurrent: boolean | null = null;
3195500cea7S猫头猫    if (targetIndex === -1) {
3205500cea7S猫头猫        // 1. 这种情况应该是出错了
3215500cea7S猫头猫        return;
3225500cea7S猫头猫    }
3235500cea7S猫头猫    // 2. 移除的是当前项
3245500cea7S猫头猫    if (currentIndex === targetIndex) {
3255500cea7S猫头猫        // 2.1 停止播放,移除当前项
3265500cea7S猫头猫        newPlayList = produce(playList, draft => {
3275500cea7S猫头猫            draft.splice(targetIndex, 1);
3285500cea7S猫头猫        });
3295500cea7S猫头猫        // 2.2 设置新的播放列表,并更新当前音乐
3305500cea7S猫头猫        if (newPlayList.length === 0) {
3315500cea7S猫头猫            currentMusic = null;
3325500cea7S猫头猫            shouldPlayCurrent = false;
3335500cea7S猫头猫        } else {
3345500cea7S猫头猫            currentMusic = newPlayList[currentIndex % newPlayList.length];
3355500cea7S猫头猫            try {
3365500cea7S猫头猫                const state = (await ReactNativeTrackPlayer.getPlaybackState())
3375500cea7S猫头猫                    .state;
3386f73e807S猫头猫                shouldPlayCurrent = !musicIsPaused(state);
3395500cea7S猫头猫            } catch {
3405500cea7S猫头猫                shouldPlayCurrent = false;
3415500cea7S猫头猫            }
3425500cea7S猫头猫        }
3435500cea7S猫头猫    } else {
3445500cea7S猫头猫        // 3. 删除
3455500cea7S猫头猫        newPlayList = produce(playList, draft => {
3465500cea7S猫头猫            draft.splice(targetIndex, 1);
3475500cea7S猫头猫        });
3485500cea7S猫头猫    }
3495500cea7S猫头猫
3505500cea7S猫头猫    setPlayList(newPlayList);
3515500cea7S猫头猫    setCurrentMusic(currentMusic);
3525500cea7S猫头猫    if (shouldPlayCurrent === true) {
3535500cea7S猫头猫        await play(currentMusic, true);
3545500cea7S猫头猫    } else if (shouldPlayCurrent === false) {
3555500cea7S猫头猫        await ReactNativeTrackPlayer.reset();
3565500cea7S猫头猫    }
3575500cea7S猫头猫};
3585500cea7S猫头猫
3595500cea7S猫头猫/**
3605500cea7S猫头猫 * 设置播放模式
3615500cea7S猫头猫 * @param mode 播放模式
3625500cea7S猫头猫 */
3635500cea7S猫头猫const setRepeatMode = (mode: MusicRepeatMode) => {
3645500cea7S猫头猫    const playList = getPlayList();
3655500cea7S猫头猫    let newPlayList;
36691eb8fa8S猫头猫    const prevMode = repeatModeStore.getValue();
36791eb8fa8S猫头猫    if (
36891eb8fa8S猫头猫        (prevMode === MusicRepeatMode.SHUFFLE &&
36991eb8fa8S猫头猫            mode !== MusicRepeatMode.SHUFFLE) ||
37091eb8fa8S猫头猫        (mode === MusicRepeatMode.SHUFFLE &&
37191eb8fa8S猫头猫            prevMode !== MusicRepeatMode.SHUFFLE)
37291eb8fa8S猫头猫    ) {
3735500cea7S猫头猫        if (mode === MusicRepeatMode.SHUFFLE) {
3745500cea7S猫头猫            newPlayList = shuffle(playList);
3755500cea7S猫头猫        } else {
37691eb8fa8S猫头猫            newPlayList = sortByTimestampAndIndex(playList, true);
37791eb8fa8S猫头猫        }
37891eb8fa8S猫头猫        setPlayList(newPlayList);
3795500cea7S猫头猫    }
3805500cea7S猫头猫
3815500cea7S猫头猫    const currentMusicItem = currentMusicStore.getValue();
3825500cea7S猫头猫    currentIndex = getMusicIndex(currentMusicItem);
3835500cea7S猫头猫    repeatModeStore.setValue(mode);
3845500cea7S猫头猫    // 更新下一首歌的信息
3855500cea7S猫头猫    ReactNativeTrackPlayer.updateMetadataForTrack(1, getFakeNextTrack());
3865500cea7S猫头猫    // 记录
387819a9075Smaotoumao    PersistStatus.set('music.repeatMode', mode);
3885500cea7S猫头猫};
3895500cea7S猫头猫
3905500cea7S猫头猫/** 清空播放列表 */
3915500cea7S猫头猫const clear = async () => {
3925500cea7S猫头猫    setPlayList([]);
3935500cea7S猫头猫    setCurrentMusic(null);
3945500cea7S猫头猫
3955500cea7S猫头猫    await ReactNativeTrackPlayer.reset();
396819a9075Smaotoumao    PersistStatus.set('music.musicItem', undefined);
397819a9075Smaotoumao    PersistStatus.set('music.progress', 0);
3985500cea7S猫头猫};
3995500cea7S猫头猫
4005500cea7S猫头猫/** 暂停 */
4015500cea7S猫头猫const pause = async () => {
4025500cea7S猫头猫    await ReactNativeTrackPlayer.pause();
4035500cea7S猫头猫};
4045500cea7S猫头猫
4058b9904b3S猫头猫/** 设置音源 */
4068b9904b3S猫头猫const setTrackSource = async (track: Track, autoPlay = true) => {
40711908939S猫头猫    if (!track.artwork?.trim()?.length) {
40811908939S猫头猫        track.artwork = undefined;
40911908939S猫头猫    }
4108b9904b3S猫头猫    await ReactNativeTrackPlayer.setQueue([track, getFakeNextTrack()]);
411819a9075Smaotoumao    PersistStatus.set('music.musicItem', track as IMusic.IMusicItem);
412819a9075Smaotoumao    PersistStatus.set('music.progress', 0);
4138b9904b3S猫头猫    if (autoPlay) {
4148b9904b3S猫头猫        await ReactNativeTrackPlayer.play();
4158b9904b3S猫头猫    }
4168b9904b3S猫头猫};
4178b9904b3S猫头猫
4185500cea7S猫头猫const setCurrentMusic = (musicItem?: IMusic.IMusicItem | null) => {
4195500cea7S猫头猫    if (!musicItem) {
4205500cea7S猫头猫        currentIndex = -1;
421f511aee9S猫头猫        currentMusicStore.setValue(null);
422819a9075Smaotoumao        PersistStatus.set('music.musicItem', undefined);
423819a9075Smaotoumao        PersistStatus.set('music.progress', 0);
4246e000b99S猫头猫        return;
4255500cea7S猫头猫    }
4265500cea7S猫头猫    currentIndex = getMusicIndex(musicItem);
4276e000b99S猫头猫    currentMusicStore.setValue(musicItem);
4285500cea7S猫头猫};
4295500cea7S猫头猫
4306e000b99S猫头猫const setQuality = (quality: IMusic.IQualityKey) => {
4316e000b99S猫头猫    qualityStore.setValue(quality);
432819a9075Smaotoumao    PersistStatus.set('music.quality', quality);
4336e000b99S猫头猫};
4345500cea7S猫头猫/**
4355500cea7S猫头猫 * 播放
4365500cea7S猫头猫 *
4375500cea7S猫头猫 * 当musicItem 为空时,代表暂停/播放
4385500cea7S猫头猫 *
4395500cea7S猫头猫 * @param musicItem
4405500cea7S猫头猫 * @param forcePlay
4415500cea7S猫头猫 * @returns
4425500cea7S猫头猫 */
4435500cea7S猫头猫const play = async (
4445500cea7S猫头猫    musicItem?: IMusic.IMusicItem | null,
4455500cea7S猫头猫    forcePlay?: boolean,
4465500cea7S猫头猫) => {
4475500cea7S猫头猫    try {
4485500cea7S猫头猫        if (!musicItem) {
4495500cea7S猫头猫            musicItem = currentMusicStore.getValue();
4505500cea7S猫头猫        }
4515500cea7S猫头猫        if (!musicItem) {
4525500cea7S猫头猫            throw new Error(PlayFailReason.PLAY_LIST_IS_EMPTY);
4535500cea7S猫头猫        }
4545500cea7S猫头猫        // 1. 移动网络禁止播放
4555500cea7S猫头猫        if (
4565500cea7S猫头猫            Network.isCellular() &&
457*41ddce91Smaotoumao            !Config.getConfig('basic.useCelluarNetworkPlay') &&
4585500cea7S猫头猫            !LocalMusicSheet.isLocalMusic(musicItem)
4595500cea7S猫头猫        ) {
4605500cea7S猫头猫            await ReactNativeTrackPlayer.reset();
4615500cea7S猫头猫            throw new Error(PlayFailReason.FORBID_CELLUAR_NETWORK_PLAY);
4625500cea7S猫头猫        }
4635500cea7S猫头猫
4645500cea7S猫头猫        // 2. 如果是当前正在播放的音频
4655500cea7S猫头猫        if (isCurrentMusic(musicItem)) {
4665500cea7S猫头猫            const currentTrack = await ReactNativeTrackPlayer.getTrack(0);
4675500cea7S猫头猫            // 2.1 如果当前有源
4685500cea7S猫头猫            if (
4695500cea7S猫头猫                currentTrack?.url &&
4705500cea7S猫头猫                isSameMediaItem(musicItem, currentTrack as IMusic.IMusicItem)
4715500cea7S猫头猫            ) {
4725500cea7S猫头猫                const currentActiveIndex =
4735500cea7S猫头猫                    await ReactNativeTrackPlayer.getActiveTrackIndex();
4745500cea7S猫头猫                if (currentActiveIndex !== 0) {
4755500cea7S猫头猫                    await ReactNativeTrackPlayer.skip(0);
4765500cea7S猫头猫                }
4775500cea7S猫头猫                if (forcePlay) {
4785500cea7S猫头猫                    // 2.1.1 强制重新开始
4795500cea7S猫头猫                    await ReactNativeTrackPlayer.seekTo(0);
4806f57784cS猫头猫                }
48166e1d5fcS猫头猫                const currentState = (
48266e1d5fcS猫头猫                    await ReactNativeTrackPlayer.getPlaybackState()
48366e1d5fcS猫头猫                ).state;
48466e1d5fcS猫头猫                if (currentState === State.Stopped) {
48566e1d5fcS猫头猫                    await setTrackSource(currentTrack);
48666e1d5fcS猫头猫                }
48766e1d5fcS猫头猫                if (currentState !== State.Playing) {
4885500cea7S猫头猫                    // 2.1.2 恢复播放
4895500cea7S猫头猫                    await ReactNativeTrackPlayer.play();
4905500cea7S猫头猫                }
4915500cea7S猫头猫                // 这种情况下,播放队列和当前歌曲都不需要变化
4925500cea7S猫头猫                return;
4935500cea7S猫头猫            }
4945500cea7S猫头猫            // 2.2 其他情况:重新获取源
4955500cea7S猫头猫        }
4965500cea7S猫头猫
4975500cea7S猫头猫        // 3. 如果没有在播放列表中,添加到队尾;同时更新列表状态
4985500cea7S猫头猫        const inPlayList = isInPlayList(musicItem);
4995500cea7S猫头猫        if (!inPlayList) {
5005500cea7S猫头猫            add(musicItem);
5015500cea7S猫头猫        }
5025500cea7S猫头猫
5035500cea7S猫头猫        // 4. 更新列表状态和当前音乐
5045500cea7S猫头猫        setCurrentMusic(musicItem);
5059cfce1b6S猫头猫        await ReactNativeTrackPlayer.reset();
5069cfce1b6S猫头猫
5079cfce1b6S猫头猫        // 4.1 刷新歌词信息
5089cfce1b6S猫头猫        if (
5099cfce1b6S猫头猫            !isSameMediaItem(
51048036ddaS猫头猫                LyricManager.getLyricState()?.lyricParser?.musicItem,
5119cfce1b6S猫头猫                musicItem,
5129cfce1b6S猫头猫            )
5139cfce1b6S猫头猫        ) {
5149cfce1b6S猫头猫            DeviceEventEmitter.emit(EDeviceEvents.REFRESH_LYRIC, true);
5159cfce1b6S猫头猫        }
5165500cea7S猫头猫
5175500cea7S猫头猫        // 5. 获取音源
5185500cea7S猫头猫        let track: IMusic.IMusicItem;
5195500cea7S猫头猫
5205500cea7S猫头猫        // 5.1 通过插件获取音源
5215500cea7S猫头猫        const plugin = PluginManager.getByName(musicItem.platform);
5225500cea7S猫头猫        // 5.2 获取音质排序
5235500cea7S猫头猫        const qualityOrder = getQualityOrder(
524*41ddce91Smaotoumao            Config.getConfig('basic.defaultPlayQuality') ?? 'standard',
525*41ddce91Smaotoumao            Config.getConfig('basic.playQualityOrder') ?? 'asc',
5265500cea7S猫头猫        );
5275500cea7S猫头猫        // 5.3 插件返回音源
5285500cea7S猫头猫        let source: IPlugin.IMediaSourceResult | null = null;
5295500cea7S猫头猫        for (let quality of qualityOrder) {
5305500cea7S猫头猫            if (isCurrentMusic(musicItem)) {
5315500cea7S猫头猫                source =
5325500cea7S猫头猫                    (await plugin?.methods?.getMediaSource(
5335500cea7S猫头猫                        musicItem,
5345500cea7S猫头猫                        quality,
5355500cea7S猫头猫                    )) ?? null;
5365500cea7S猫头猫                // 5.3.1 获取到真实源
5375500cea7S猫头猫                if (source) {
5386e000b99S猫头猫                    setQuality(quality);
5395500cea7S猫头猫                    break;
5405500cea7S猫头猫                }
5415500cea7S猫头猫            } else {
5425500cea7S猫头猫                // 5.3.2 已经切换到其他歌曲了,
5435500cea7S猫头猫                return;
5445500cea7S猫头猫            }
5455500cea7S猫头猫        }
5465500cea7S猫头猫
5475500cea7S猫头猫        if (!isCurrentMusic(musicItem)) {
5485500cea7S猫头猫            return;
5495500cea7S猫头猫        }
5505500cea7S猫头猫        if (!source) {
55143eb30bfS猫头猫            // 如果有source
55243eb30bfS猫头猫            if (musicItem.source) {
55343eb30bfS猫头猫                for (let quality of qualityOrder) {
55443eb30bfS猫头猫                    if (musicItem.source[quality]?.url) {
55543eb30bfS猫头猫                        source = musicItem.source[quality]!;
5566e000b99S猫头猫                        setQuality(quality);
5576e000b99S猫头猫
55843eb30bfS猫头猫                        break;
5595500cea7S猫头猫                    }
56043eb30bfS猫头猫                }
56143eb30bfS猫头猫            }
56243eb30bfS猫头猫            // 5.4 没有返回源
56343eb30bfS猫头猫            if (!source && !musicItem.url) {
5643991724eS猫头猫                // 插件失效的情况
565*41ddce91Smaotoumao                if (Config.getConfig('basic.tryChangeSourceWhenPlayFail')) {
5663991724eS猫头猫                    // 重试
5673991724eS猫头猫                    const similarMusic = await getSimilarMusic(
5683991724eS猫头猫                        musicItem,
5693991724eS猫头猫                        'music',
5703991724eS猫头猫                        () => !isCurrentMusic(musicItem),
5713991724eS猫头猫                    );
5723991724eS猫头猫
5733991724eS猫头猫                    if (similarMusic) {
5743991724eS猫头猫                        const similarMusicPlugin =
5753991724eS猫头猫                            PluginManager.getByMedia(similarMusic);
5763991724eS猫头猫
5773991724eS猫头猫                        for (let quality of qualityOrder) {
5783991724eS猫头猫                            if (isCurrentMusic(musicItem)) {
5793991724eS猫头猫                                source =
5803991724eS猫头猫                                    (await similarMusicPlugin?.methods?.getMediaSource(
5813991724eS猫头猫                                        similarMusic,
5823991724eS猫头猫                                        quality,
5833991724eS猫头猫                                    )) ?? null;
5843991724eS猫头猫                                // 5.4.1 获取到真实源
5853991724eS猫头猫                                if (source) {
5863991724eS猫头猫                                    setQuality(quality);
5873991724eS猫头猫                                    break;
5883991724eS猫头猫                                }
5893991724eS猫头猫                            } else {
5903991724eS猫头猫                                // 5.4.2 已经切换到其他歌曲了,
5913991724eS猫头猫                                return;
5923991724eS猫头猫                            }
5933991724eS猫头猫                        }
5943991724eS猫头猫                    }
5953991724eS猫头猫
5963991724eS猫头猫                    if (!source) {
59743eb30bfS猫头猫                        throw new Error(PlayFailReason.INVALID_SOURCE);
5983991724eS猫头猫                    }
5993991724eS猫头猫                } else {
6003991724eS猫头猫                    throw new Error(PlayFailReason.INVALID_SOURCE);
6013991724eS猫头猫                }
60243eb30bfS猫头猫            } else {
6035500cea7S猫头猫                source = {
6045500cea7S猫头猫                    url: musicItem.url,
6055500cea7S猫头猫                };
6066e000b99S猫头猫                setQuality('standard');
6075500cea7S猫头猫            }
60843eb30bfS猫头猫        }
6095500cea7S猫头猫
6105500cea7S猫头猫        // 6. 特殊类型源
6115500cea7S猫头猫        if (getUrlExt(source.url) === '.m3u8') {
6125500cea7S猫头猫            // @ts-ignore
6135500cea7S猫头猫            source.type = 'hls';
6145500cea7S猫头猫        }
6155500cea7S猫头猫        // 7. 合并结果
6165500cea7S猫头猫        track = mergeProps(musicItem, source) as IMusic.IMusicItem;
6175500cea7S猫头猫
6185500cea7S猫头猫        // 8. 新增历史记录
6195500cea7S猫头猫        musicHistory.addMusic(musicItem);
6205500cea7S猫头猫
62162e73a5eS猫头猫        trace('获取音源成功', track);
6225500cea7S猫头猫        // 9. 设置音源
6235500cea7S猫头猫        await setTrackSource(track as Track);
6245500cea7S猫头猫
6255500cea7S猫头猫        // 10. 获取补充信息
6265500cea7S猫头猫        let info: Partial<IMusic.IMusicItem> | null = null;
6275500cea7S猫头猫        try {
6285500cea7S猫头猫            info = (await plugin?.methods?.getMusicInfo?.(musicItem)) ?? null;
62963d241f1S猫头猫            if (
63063d241f1S猫头猫                (typeof info?.url === 'string' && info.url.trim() === '') ||
63163d241f1S猫头猫                (info?.url && typeof info.url !== 'string')
63263d241f1S猫头猫            ) {
633f9c53a4cS猫头猫                delete info.url;
634f9c53a4cS猫头猫            }
6355500cea7S猫头猫        } catch {}
6365500cea7S猫头猫
6375500cea7S猫头猫        // 11. 设置补充信息
6385500cea7S猫头猫        if (info && isCurrentMusic(musicItem)) {
6395500cea7S猫头猫            const mergedTrack = mergeProps(track, info);
6405500cea7S猫头猫            currentMusicStore.setValue(mergedTrack as IMusic.IMusicItem);
6415500cea7S猫头猫            await ReactNativeTrackPlayer.updateMetadataForTrack(
6425500cea7S猫头猫                0,
6435500cea7S猫头猫                mergedTrack as TrackMetadataBase,
6445500cea7S猫头猫            );
6455500cea7S猫头猫        }
6465500cea7S猫头猫    } catch (e: any) {
6475500cea7S猫头猫        const message = e?.message;
6485500cea7S猫头猫        if (
6495500cea7S猫头猫            message === 'The player is not initialized. Call setupPlayer first.'
6505500cea7S猫头猫        ) {
6515500cea7S猫头猫            await ReactNativeTrackPlayer.setupPlayer();
6525500cea7S猫头猫            play(musicItem, forcePlay);
6535500cea7S猫头猫        } else if (message === PlayFailReason.FORBID_CELLUAR_NETWORK_PLAY) {
654dffbbaffS猫头猫            if (getCurrentDialog()?.name !== 'SimpleDialog') {
655595a35deS猫头猫                showDialog('SimpleDialog', {
656595a35deS猫头猫                    title: '流量提醒',
657595a35deS猫头猫                    content:
6582dd57e49S猫头猫                        '当前非WIFI环境,侧边栏设置中打开【使用移动网络播放】功能后可继续播放',
659595a35deS猫头猫                });
660dffbbaffS猫头猫            }
6615500cea7S猫头猫        } else if (message === PlayFailReason.INVALID_SOURCE) {
66262e73a5eS猫头猫            trace('音源为空,播放失败');
6636f57784cS猫头猫            await failToPlay();
6645500cea7S猫头猫        } else if (message === PlayFailReason.PLAY_LIST_IS_EMPTY) {
6655500cea7S猫头猫            // 队列是空的,不应该出现这种情况
6665500cea7S猫头猫        }
6675500cea7S猫头猫    }
6685500cea7S猫头猫};
6695500cea7S猫头猫
6705500cea7S猫头猫/**
6715500cea7S猫头猫 * 播放音乐,同时替换播放队列
6725500cea7S猫头猫 * @param musicItem 音乐
6735500cea7S猫头猫 * @param newPlayList 替代列表
6745500cea7S猫头猫 */
6755500cea7S猫头猫const playWithReplacePlayList = async (
6765500cea7S猫头猫    musicItem: IMusic.IMusicItem,
6775500cea7S猫头猫    newPlayList: IMusic.IMusicItem[],
6785500cea7S猫头猫) => {
6795500cea7S猫头猫    if (newPlayList.length !== 0) {
6805500cea7S猫头猫        const now = Date.now();
6815500cea7S猫头猫        if (newPlayList.length > maxMusicQueueLength) {
6825500cea7S猫头猫            newPlayList = shrinkPlayListToSize(
6835500cea7S猫头猫                newPlayList,
6845500cea7S猫头猫                newPlayList.findIndex(it => isSameMediaItem(it, musicItem)),
6855500cea7S猫头猫            );
6865500cea7S猫头猫        }
6874c222b69S猫头猫
6884c222b69S猫头猫        newPlayList.forEach((it, index) => {
6894c222b69S猫头猫            it[timeStampSymbol] = now;
6904c222b69S猫头猫            it[sortIndexSymbol] = index;
6914c222b69S猫头猫        });
6924c222b69S猫头猫
6935500cea7S猫头猫        setPlayList(
6945500cea7S猫头猫            repeatModeStore.getValue() === MusicRepeatMode.SHUFFLE
6954c222b69S猫头猫                ? shuffle(newPlayList)
6964c222b69S猫头猫                : newPlayList,
6975500cea7S猫头猫        );
6985500cea7S猫头猫        await play(musicItem, true);
6995500cea7S猫头猫    }
7005500cea7S猫头猫};
7015500cea7S猫头猫
7026f57784cS猫头猫const skipToNext = async () => {
7035500cea7S猫头猫    if (isPlayListEmpty()) {
7045500cea7S猫头猫        setCurrentMusic(null);
7055500cea7S猫头猫        return;
7065500cea7S猫头猫    }
7075500cea7S猫头猫
7085500cea7S猫头猫    await play(getPlayListMusicAt(currentIndex + 1), true);
7095500cea7S猫头猫};
7105500cea7S猫头猫
7115500cea7S猫头猫const skipToPrevious = async () => {
7125500cea7S猫头猫    if (isPlayListEmpty()) {
7135500cea7S猫头猫        setCurrentMusic(null);
7145500cea7S猫头猫        return;
7155500cea7S猫头猫    }
7165500cea7S猫头猫
7176f57784cS猫头猫    await play(
7186f57784cS猫头猫        getPlayListMusicAt(currentIndex === -1 ? 0 : currentIndex - 1),
7196f57784cS猫头猫        true,
7206f57784cS猫头猫    );
7215500cea7S猫头猫};
7225500cea7S猫头猫
7235500cea7S猫头猫/** 修改当前播放的音质 */
7245500cea7S猫头猫const changeQuality = async (newQuality: IMusic.IQualityKey) => {
7255500cea7S猫头猫    // 获取当前的音乐和进度
7265500cea7S猫头猫    if (newQuality === qualityStore.getValue()) {
7275500cea7S猫头猫        return true;
7285500cea7S猫头猫    }
7295500cea7S猫头猫
7305500cea7S猫头猫    // 获取当前歌曲
7315500cea7S猫头猫    const musicItem = currentMusicStore.getValue();
7325500cea7S猫头猫    if (!musicItem) {
7335500cea7S猫头猫        return false;
7345500cea7S猫头猫    }
7355500cea7S猫头猫    try {
7365500cea7S猫头猫        const progress = await ReactNativeTrackPlayer.getProgress();
7375500cea7S猫头猫        const plugin = PluginManager.getByMedia(musicItem);
7385500cea7S猫头猫        const newSource = await plugin?.methods?.getMediaSource(
7395500cea7S猫头猫            musicItem,
7405500cea7S猫头猫            newQuality,
7415500cea7S猫头猫        );
7425500cea7S猫头猫        if (!newSource?.url) {
7435500cea7S猫头猫            throw new Error(PlayFailReason.INVALID_SOURCE);
7445500cea7S猫头猫        }
7455500cea7S猫头猫        if (isCurrentMusic(musicItem)) {
7465500cea7S猫头猫            const playingState = (
7475500cea7S猫头猫                await ReactNativeTrackPlayer.getPlaybackState()
7485500cea7S猫头猫            ).state;
7495500cea7S猫头猫            await setTrackSource(
7505500cea7S猫头猫                mergeProps(musicItem, newSource) as unknown as Track,
7515500cea7S猫头猫                !musicIsPaused(playingState),
7525500cea7S猫头猫            );
7535500cea7S猫头猫
7545500cea7S猫头猫            await ReactNativeTrackPlayer.seekTo(progress.position ?? 0);
7556e000b99S猫头猫            setQuality(newQuality);
7565500cea7S猫头猫        }
7575500cea7S猫头猫        return true;
7585500cea7S猫头猫    } catch {
7595500cea7S猫头猫        // 修改失败
7605500cea7S猫头猫        return false;
7615500cea7S猫头猫    }
7625500cea7S猫头猫};
7635500cea7S猫头猫
7645500cea7S猫头猫enum PlayFailReason {
7655500cea7S猫头猫    /** 禁止移动网络播放 */
7665500cea7S猫头猫    FORBID_CELLUAR_NETWORK_PLAY = 'FORBID_CELLUAR_NETWORK_PLAY',
7675500cea7S猫头猫    /** 播放列表为空 */
7685500cea7S猫头猫    PLAY_LIST_IS_EMPTY = 'PLAY_LIST_IS_EMPTY',
7695500cea7S猫头猫    /** 无效源 */
7705500cea7S猫头猫    INVALID_SOURCE = 'INVALID_SOURCE',
7715500cea7S猫头猫    /** 非当前音乐 */
7725500cea7S猫头猫}
7735500cea7S猫头猫
7745500cea7S猫头猫function useMusicState() {
7755500cea7S猫头猫    const playbackState = usePlaybackState();
7765500cea7S猫头猫
7775500cea7S猫头猫    return playbackState.state;
7785500cea7S猫头猫}
7795500cea7S猫头猫
780f511aee9S猫头猫function getPreviousMusic() {
781f511aee9S猫头猫    const currentMusicItem = currentMusicStore.getValue();
782f511aee9S猫头猫    if (!currentMusicItem) {
783f511aee9S猫头猫        return null;
784f511aee9S猫头猫    }
785f511aee9S猫头猫
786f511aee9S猫头猫    return getPlayListMusicAt(currentIndex - 1);
787f511aee9S猫头猫}
788f511aee9S猫头猫
789f511aee9S猫头猫function getNextMusic() {
790f511aee9S猫头猫    const currentMusicItem = currentMusicStore.getValue();
791f511aee9S猫头猫    if (!currentMusicItem) {
792f511aee9S猫头猫        return null;
793f511aee9S猫头猫    }
794f511aee9S猫头猫
795f511aee9S猫头猫    return getPlayListMusicAt(currentIndex + 1);
796f511aee9S猫头猫}
797f511aee9S猫头猫
7985500cea7S猫头猫const TrackPlayer = {
7995500cea7S猫头猫    setupTrackPlayer,
8005500cea7S猫头猫    usePlayList,
8015500cea7S猫头猫    getPlayList,
8025500cea7S猫头猫    addAll,
8035500cea7S猫头猫    add,
8045500cea7S猫头猫    addNext,
8055500cea7S猫头猫    skipToNext,
8065500cea7S猫头猫    skipToPrevious,
8075500cea7S猫头猫    play,
8085500cea7S猫头猫    playWithReplacePlayList,
8095500cea7S猫头猫    pause,
8105500cea7S猫头猫    remove,
8115500cea7S猫头猫    clear,
8125500cea7S猫头猫    useCurrentMusic: currentMusicStore.useValue,
8135500cea7S猫头猫    getCurrentMusic: currentMusicStore.getValue,
8145500cea7S猫头猫    useRepeatMode: repeatModeStore.useValue,
8155500cea7S猫头猫    getRepeatMode: repeatModeStore.getValue,
8165500cea7S猫头猫    toggleRepeatMode,
8175500cea7S猫头猫    usePlaybackState,
8185500cea7S猫头猫    getProgress: ReactNativeTrackPlayer.getProgress,
8195500cea7S猫头猫    useProgress: useProgress,
8205500cea7S猫头猫    seekTo: ReactNativeTrackPlayer.seekTo,
8215500cea7S猫头猫    changeQuality,
8225500cea7S猫头猫    useCurrentQuality: qualityStore.useValue,
8235500cea7S猫头猫    getCurrentQuality: qualityStore.getValue,
8245500cea7S猫头猫    getRate: ReactNativeTrackPlayer.getRate,
8255500cea7S猫头猫    setRate: ReactNativeTrackPlayer.setRate,
8265500cea7S猫头猫    useMusicState,
8275500cea7S猫头猫    reset: ReactNativeTrackPlayer.reset,
828f511aee9S猫头猫    getPreviousMusic,
829f511aee9S猫头猫    getNextMusic,
8305500cea7S猫头猫};
8315500cea7S猫头猫
8325500cea7S猫头猫export default TrackPlayer;
8335500cea7S猫头猫export {MusicRepeatMode, State as MusicState};
834