refactor: use dexie.js instead of localforage

This commit is contained in:
qier222 2021-03-25 19:28:40 +08:00
parent bf61fe6e28
commit fee97f7f3c
No known key found for this signature in database
GPG key ID: 9C85007ED905F14D
6 changed files with 69 additions and 55 deletions

View file

@ -1,8 +1,7 @@
import { getTrackDetail, scrobble, getMP3 } from "@/api/track";
import { shuffle } from "lodash";
import { Howler, Howl } from "howler";
import localforage from "localforage";
import { cacheTrack } from "@/utils/db";
import { cacheTrackSource, getTrackSource } from "@/utils/db";
import { getAlbum } from "@/api/album";
import { getPlaylistDetail } from "@/api/playlist";
import { getArtist } from "@/api/artist";
@ -201,10 +200,9 @@ export default class {
});
}
_getAudioSourceFromCache(id) {
let tracks = localforage.createInstance({ name: "tracks" });
return tracks.getItem(id).then((t) => {
if (t === null) return null;
const source = URL.createObjectURL(new Blob([t.mp3]));
return getTrackSource(id).then((t) => {
if (!t) return null;
const source = URL.createObjectURL(new Blob([t.source]));
return source;
});
}
@ -216,7 +214,7 @@ export default class {
if (result.data[0].freeTrialInfo !== null) return null; // 跳过只能试听的歌曲
const source = result.data[0].url.replace(/^http:/, "https:");
if (store.state.settings.automaticallyCacheSongs) {
cacheTrack(track.id, source);
cacheTrackSource(track, source, result.data[0].br);
}
return source;
});
@ -230,7 +228,8 @@ export default class {
if (process.env.IS_ELECTRON !== true) return null;
const source = ipcRenderer.sendSync("unblock-music", track);
if (store.state.settings.automaticallyCacheSongs && source?.url) {
cacheTrack(track.id, source.url);
// TODO: 将unblockMusic字样换成真正的来源比如酷我咪咕等
cacheTrackSource(track, source.url, 128000, "unblockMusic");
}
return source?.url;
}

View file

@ -1,54 +1,66 @@
import axios from "axios";
import localforage from "localforage";
import Dexie from "dexie";
// import pkg from "../../package.json";
export function cacheTrack(id, url) {
let tracks = localforage.createInstance({
name: "tracks",
});
const db = new Dexie("yesplaymusic");
db.version(1).stores({
trackSources: "&id",
});
// TODO: limit cache songs number
// tracks.length().then(function (length) {
// if (length > 2) {
// tracks.keys().then(function (keys) {
// tracks.removeItem(keys[keys.length - 2]);
// });
// }
// });
// TODO: cache track details
export function cacheTrackSource(trackInfo, url, bitRate, from = "netease") {
const name = trackInfo.name;
const artist = trackInfo.ar[0]?.name || trackInfo.artists[0]?.name;
return axios
.get(url, {
responseType: "arraybuffer",
})
.then((response) => {
tracks.setItem(`${id}`, { mp3: response.data });
return { mp3: response.data };
db.trackSources.put({
id: trackInfo.id,
source: response.data,
bitRate,
from,
name,
artist,
});
console.debug(`[debug][db.js] cached track 👉 ${name} by ${artist}`);
return { trackID: trackInfo.id, source: response.data, bitRate };
});
}
export function countDBSize(dbName) {
let db = localforage.createInstance({
name: dbName,
export function getTrackSource(id) {
return db.trackSources.get(Number(id)).then((track) => {
if (!track) return null;
console.debug(
`[debug][db.js] get track from cache 👉 ${track.name} by ${track.artist}`
);
return track;
});
}
export function countDBSize() {
let trackSizes = [];
return db
.iterate((value) => {
trackSizes.push(value.mp3.byteLength);
return db.trackSources
.each((track) => {
trackSizes.push(track.source.byteLength);
})
.then(() => {
return {
bytes: trackSizes.reduce((s1, s2) => s1 + s2, 0),
length: trackSizes.length,
};
})
.catch((err) => {
console.log(err);
});
}
export function clearDB(dbName) {
let db = localforage.createInstance({
name: dbName,
export function clearDB() {
return new Promise((resolve) => {
db.tables.forEach(function (table) {
table.clear();
});
resolve();
});
return db.clear();
}
window.cacheTrackSource = cacheTrackSource;
window.db = db;
window.countDBSize = countDBSize;

View file

@ -54,9 +54,15 @@ const updatePlayer = () => {
localStorage.setItem("player", JSON.stringify(data));
};
const removeOldStuff = () => {
// remove old indexedDB databases created by localforage
indexedDB.deleteDatabase("tracks");
};
export default function () {
updateSetting();
updateData();
updatePlayer();
removeOldStuff();
localStorage.setItem("appVersion", JSON.stringify(pkg.version));
}

View file

@ -122,7 +122,7 @@
>
</div>
<div class="right">
<button @click="clearCache('tracks')">
<button @click="clearCache()">
{{ $t("settings.clearSongsCache") }}
</button>
</div>
@ -365,7 +365,7 @@ export default {
set(value) {
if (value === this.settings.musicQuality) return;
this.$store.commit("changeMusicQuality", value);
this.clearCache("tracks");
this.clearCache();
},
},
lyricFontSize: {
@ -443,7 +443,7 @@ export default {
value,
});
if (value === false) {
this.clearCache("tracks");
this.clearCache();
}
},
},
@ -520,8 +520,8 @@ export default {
doLogout();
this.$router.push({ name: "home" });
},
countDBSize(dbName) {
countDBSize(dbName).then((data) => {
countDBSize() {
countDBSize().then((data) => {
if (data === undefined) {
this.tracksCache = {
size: "0KB",
@ -533,10 +533,9 @@ export default {
this.tracksCache.length = data.length;
});
},
clearCache(dbName) {
// TODO: toast
clearDB(dbName).then(() => {
this.countDBSize("tracks");
clearCache() {
clearDB().then(() => {
this.countDBSize();
});
},
lastfmConnect() {