2023-01-13 05:40:33 +01:00
|
|
|
import { markRaw, ref } from "vue";
|
2024-04-17 22:34:23 +02:00
|
|
|
import type { ApiTypes, entities } from "firefish-js";
|
2024-04-22 04:32:41 +02:00
|
|
|
import { isSignedIn, me } from "./me";
|
2024-03-16 16:49:12 +01:00
|
|
|
import { Storage } from "./pizzax";
|
2024-04-11 15:17:49 +02:00
|
|
|
import type { NoteVisibility } from "@/types/note";
|
2020-02-09 23:23:43 +01:00
|
|
|
|
2024-04-12 10:37:32 +02:00
|
|
|
export const postFormActions: {
|
|
|
|
title: string;
|
2024-04-13 17:08:58 +02:00
|
|
|
handler: (from, update) => void | Promise<void>;
|
2024-04-12 10:37:32 +02:00
|
|
|
}[] = [];
|
|
|
|
export const userActions: {
|
|
|
|
title: string;
|
2024-04-13 17:08:58 +02:00
|
|
|
handler: (user: entities.User) => void | Promise<void>;
|
2024-04-12 10:37:32 +02:00
|
|
|
}[] = [];
|
|
|
|
export const noteActions: {
|
|
|
|
title: string;
|
|
|
|
handler: (note: entities.Note) => void | Promise<void>;
|
|
|
|
}[] = [];
|
|
|
|
export const noteViewInterruptors: {
|
|
|
|
handler: (note: entities.Note) => Promise<entities.Note>;
|
|
|
|
}[] = [];
|
|
|
|
export const notePostInterruptors: {
|
2024-04-13 17:08:58 +02:00
|
|
|
handler: (note: ApiTypes.NoteSubmitReq) => Promise<ApiTypes.NoteSubmitReq>;
|
2024-04-12 10:37:32 +02:00
|
|
|
}[] = [];
|
2020-10-17 13:12:00 +02:00
|
|
|
|
2023-04-07 08:02:07 +02:00
|
|
|
const menuOptions = [
|
|
|
|
"notifications",
|
|
|
|
"followRequests",
|
|
|
|
"messaging",
|
|
|
|
"explore",
|
|
|
|
"favorites",
|
|
|
|
"channels",
|
2023-04-08 01:47:04 +02:00
|
|
|
"search",
|
2023-04-07 08:02:07 +02:00
|
|
|
];
|
|
|
|
|
2023-09-21 05:46:48 +02:00
|
|
|
export const defaultReactions = [
|
|
|
|
"⭐",
|
|
|
|
"❤️",
|
|
|
|
"😆",
|
|
|
|
"🤔",
|
|
|
|
"😮",
|
|
|
|
"🎉",
|
|
|
|
"💢",
|
|
|
|
"😥",
|
|
|
|
"😇",
|
|
|
|
"🥴",
|
2023-09-21 06:49:12 +02:00
|
|
|
"🔥",
|
|
|
|
"🐟",
|
2023-09-21 05:46:48 +02:00
|
|
|
];
|
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
// TODO: それぞれいちいちwhereとかdefaultというキーを付けなきゃいけないの冗長なのでなんとかする(ただ型定義が面倒になりそう)
|
|
|
|
// あと、現行の定義の仕方なら「whereが何であるかに関わらずキー名の重複不可」という制約を付けられるメリットもあるからそのメリットを引き継ぐ方法も考えないといけない
|
2023-01-13 05:40:33 +01:00
|
|
|
export const defaultStore = markRaw(
|
|
|
|
new Storage("base", {
|
|
|
|
tutorial: {
|
|
|
|
where: "account",
|
|
|
|
default: 0,
|
|
|
|
},
|
2023-05-29 20:51:02 +02:00
|
|
|
tlHomeHintClosed: {
|
2023-10-15 19:04:44 +02:00
|
|
|
where: "account",
|
2023-05-29 20:51:02 +02:00
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
tlLocalHintClosed: {
|
2023-10-15 19:04:44 +02:00
|
|
|
where: "account",
|
2023-05-29 20:51:02 +02:00
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
tlRecommendedHintClosed: {
|
2023-10-15 19:04:44 +02:00
|
|
|
where: "account",
|
2023-05-29 20:51:02 +02:00
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
tlSocialHintClosed: {
|
2023-10-15 19:04:44 +02:00
|
|
|
where: "account",
|
2023-05-29 20:51:02 +02:00
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
tlGlobalHintClosed: {
|
2023-10-15 19:04:44 +02:00
|
|
|
where: "account",
|
2023-05-29 20:51:02 +02:00
|
|
|
default: false,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
keepCw: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
showFullAcct: {
|
|
|
|
where: "account",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
rememberNoteVisibility: {
|
|
|
|
where: "account",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
defaultNoteVisibility: {
|
|
|
|
where: "account",
|
2024-04-09 19:18:08 +02:00
|
|
|
default: "public" as NoteVisibility,
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
|
|
|
defaultNoteLocalOnly: {
|
|
|
|
where: "account",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
uploadFolder: {
|
|
|
|
where: "account",
|
|
|
|
default: null as string | null,
|
|
|
|
},
|
|
|
|
pastedFileName: {
|
|
|
|
where: "account",
|
|
|
|
default: "yyyy-MM-dd HH-mm-ss [{{number}}]",
|
|
|
|
},
|
|
|
|
keepOriginalUploading: {
|
|
|
|
where: "account",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
memo: {
|
|
|
|
where: "account",
|
|
|
|
default: null,
|
|
|
|
},
|
|
|
|
reactions: {
|
|
|
|
where: "account",
|
2023-09-21 05:46:48 +02:00
|
|
|
default: defaultReactions,
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
|
|
|
mutedWords: {
|
|
|
|
where: "account",
|
|
|
|
default: [],
|
|
|
|
},
|
2023-09-20 02:07:41 +02:00
|
|
|
mutedLangs: {
|
|
|
|
where: "account",
|
|
|
|
default: [],
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
mutedAds: {
|
|
|
|
where: "account",
|
|
|
|
default: [] as string[],
|
|
|
|
},
|
|
|
|
showAds: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
menu: {
|
|
|
|
where: "deviceAccount",
|
2023-04-07 08:02:07 +02:00
|
|
|
default: menuOptions,
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
|
|
|
visibility: {
|
|
|
|
where: "deviceAccount",
|
2024-04-09 19:18:08 +02:00
|
|
|
default: "public" as NoteVisibility,
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
|
|
|
localOnly: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
statusbars: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: [] as {
|
|
|
|
name: string;
|
|
|
|
id: string;
|
|
|
|
type: string;
|
|
|
|
size: "verySmall" | "small" | "medium" | "large" | "veryLarge";
|
|
|
|
black: boolean;
|
|
|
|
props: Record<string, any>;
|
|
|
|
}[],
|
|
|
|
},
|
|
|
|
widgets: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: [] as {
|
|
|
|
name: string;
|
|
|
|
id: string;
|
|
|
|
place: string | null;
|
|
|
|
data: Record<string, any>;
|
|
|
|
}[],
|
|
|
|
},
|
|
|
|
tl: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: {
|
2024-04-22 04:32:41 +02:00
|
|
|
src: (isSignedIn(me) ? "home" : "local") as
|
2024-03-01 16:19:05 +01:00
|
|
|
| "home"
|
|
|
|
| "local"
|
|
|
|
| "social"
|
|
|
|
| "global"
|
|
|
|
| "recommended",
|
2023-01-13 05:40:33 +01:00
|
|
|
arg: null,
|
|
|
|
},
|
|
|
|
},
|
2020-12-05 04:50:09 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
overridedDeviceKind: {
|
|
|
|
where: "device",
|
|
|
|
default: null as null | "smartphone" | "tablet" | "desktop",
|
|
|
|
},
|
|
|
|
serverDisconnectedBehavior: {
|
|
|
|
where: "device",
|
|
|
|
default: "nothing" as "nothing" | "quiet" | "reload" | "dialog",
|
|
|
|
},
|
|
|
|
seperateRenoteQuote: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
2023-05-07 01:26:24 +02:00
|
|
|
},
|
2023-06-01 05:35:25 +02:00
|
|
|
expandOnNoteClick: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
nsfw: {
|
2024-02-15 19:55:31 +01:00
|
|
|
where: "deviceAccount",
|
2023-01-13 05:40:33 +01:00
|
|
|
default: "respect" as "respect" | "force" | "ignore",
|
|
|
|
},
|
|
|
|
animation: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-06-03 23:12:08 +02:00
|
|
|
advancedMfm: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
animatedMfm: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-05-13 03:48:58 +02:00
|
|
|
animatedMfmWarnShown: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
loadRawImages: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
imageNewTab: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
disableShowingAnimatedImages: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
disablePagesScript: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
useOsNativeEmojis: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
disableDrawer: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
useBlurEffectForModal: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
useBlurEffect: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
showFixedPostForm: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
enableInfiniteScroll: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
useReactionPickerForContextMenu: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
showGapBetweenNotesInTimeline: {
|
|
|
|
where: "device",
|
2024-03-01 00:51:19 +01:00
|
|
|
default: false,
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
|
|
|
darkMode: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
instanceTicker: {
|
|
|
|
where: "device",
|
2024-03-01 00:51:19 +01:00
|
|
|
default: "always" as "none" | "remote" | "always",
|
2023-01-13 05:40:33 +01:00
|
|
|
},
|
2023-06-23 05:58:44 +02:00
|
|
|
reactionPickerSkinTone: {
|
|
|
|
where: "account",
|
|
|
|
default: 1,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
reactionPickerSize: {
|
|
|
|
where: "device",
|
|
|
|
default: 3,
|
|
|
|
},
|
|
|
|
reactionPickerWidth: {
|
|
|
|
where: "device",
|
|
|
|
default: 3,
|
|
|
|
},
|
|
|
|
reactionPickerHeight: {
|
|
|
|
where: "device",
|
|
|
|
default: 3,
|
|
|
|
},
|
|
|
|
reactionPickerUseDrawerForMobile: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
recentlyUsedEmojis: {
|
|
|
|
where: "device",
|
|
|
|
default: [] as string[],
|
|
|
|
},
|
|
|
|
recentlyUsedUsers: {
|
|
|
|
where: "device",
|
|
|
|
default: [] as string[],
|
|
|
|
},
|
|
|
|
defaultSideView: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
menuDisplay: {
|
|
|
|
where: "device",
|
|
|
|
default: "sideFull" as "sideFull" | "sideIcon" | "top",
|
|
|
|
},
|
|
|
|
reportError: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
squareAvatars: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2024-02-15 05:10:24 +01:00
|
|
|
squareCatAvatars: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
postFormWithHashtags: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
postFormHashtags: {
|
|
|
|
where: "device",
|
|
|
|
default: "",
|
|
|
|
},
|
|
|
|
themeInitial: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
numberOfPageCache: {
|
|
|
|
where: "device",
|
|
|
|
default: 5,
|
|
|
|
},
|
|
|
|
enterSendsMessage: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
showUpdates: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
swipeOnDesktop: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
2023-06-26 20:36:28 +02:00
|
|
|
swipeOnMobile: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
showAdminUpdates: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
woozyMode: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
2023-03-30 05:11:57 +02:00
|
|
|
enableCustomKaTeXMacro: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
2023-04-20 04:53:28 +02:00
|
|
|
enableEmojiReactions: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-04-20 09:57:36 +02:00
|
|
|
showEmojisInReactionNotifications: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-06-15 05:17:56 +02:00
|
|
|
showTimelineReplies: {
|
2023-09-09 22:16:23 +02:00
|
|
|
where: "deviceAccount",
|
2023-06-27 14:21:28 +02:00
|
|
|
default: false,
|
2023-06-24 06:22:44 +02:00
|
|
|
},
|
2023-07-21 15:48:27 +02:00
|
|
|
addRe: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-09-03 22:50:15 +02:00
|
|
|
detectPostLanguage: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-09-17 05:23:20 +02:00
|
|
|
openServerInfo: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2023-10-17 03:57:20 +02:00
|
|
|
iconSet: {
|
|
|
|
where: "device",
|
|
|
|
default: "ph-bold" as
|
|
|
|
| "ph-bold"
|
|
|
|
| "ph-duotone"
|
|
|
|
| "ph-light"
|
|
|
|
| "ph" // this is ph-regular
|
|
|
|
| "ph-fill",
|
|
|
|
},
|
2023-11-23 09:49:13 +01:00
|
|
|
recentlyUsedPostLanguages: {
|
|
|
|
where: "account",
|
|
|
|
default: [] as string[],
|
|
|
|
},
|
2023-12-05 04:50:44 +01:00
|
|
|
useEmojiCdn: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2024-02-10 18:08:25 +01:00
|
|
|
showPreviewByDefault: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: false,
|
|
|
|
},
|
2024-02-10 18:29:32 +01:00
|
|
|
hideFollowButtons: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2024-02-15 05:35:26 +01:00
|
|
|
replaceChatButtonWithAccountButton: {
|
|
|
|
where: "device",
|
|
|
|
default: true,
|
|
|
|
},
|
2024-02-15 05:43:23 +01:00
|
|
|
replaceWidgetsButtonWithReloadButton: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
2024-02-15 16:35:19 +01:00
|
|
|
searchURL: {
|
|
|
|
where: "device",
|
|
|
|
default: "https://duckduckgo.com/?q=",
|
|
|
|
},
|
2024-02-15 20:14:19 +01:00
|
|
|
showBigPostButton: {
|
|
|
|
where: "device",
|
|
|
|
default: false,
|
|
|
|
},
|
2024-02-29 14:21:19 +01:00
|
|
|
enableTimelineStreaming: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
enablePullToRefresh: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
pullToRefreshThreshold: {
|
|
|
|
where: "device",
|
|
|
|
default: 150,
|
|
|
|
},
|
2024-03-06 06:34:29 +01:00
|
|
|
showNoAltTextWarning: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
|
|
|
},
|
2024-03-25 15:52:37 +01:00
|
|
|
showAddFileDescriptionAtFirstPost: {
|
|
|
|
where: "account",
|
|
|
|
default: false,
|
|
|
|
},
|
2024-03-19 18:08:14 +01:00
|
|
|
autocorrectNoteLanguage: {
|
|
|
|
where: "account",
|
|
|
|
default: true,
|
2024-03-19 19:07:55 +01:00
|
|
|
},
|
2024-04-24 15:33:56 +02:00
|
|
|
foldNotification: {
|
2024-04-25 02:55:18 +02:00
|
|
|
where: "deviceAccount",
|
2024-04-27 07:18:01 +02:00
|
|
|
default: true,
|
2024-04-24 15:33:56 +02:00
|
|
|
},
|
2024-05-02 18:22:25 +02:00
|
|
|
mergeThreadInTimeline: {
|
2024-05-01 18:53:52 +02:00
|
|
|
where: "deviceAccount",
|
2024-05-02 18:22:25 +02:00
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
mergeRenotesInTimeline: {
|
|
|
|
where: "deviceAccount",
|
|
|
|
default: true,
|
2024-05-01 18:53:52 +02:00
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
}),
|
|
|
|
);
|
2020-07-11 17:14:34 +02:00
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
// TODO: 他のタブと永続化されたstateを同期
|
2020-07-11 17:38:55 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
const PREFIX = "miux:";
|
2020-07-11 17:38:55 +02:00
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
interface Plugin {
|
2020-12-19 02:55:52 +01:00
|
|
|
id: string;
|
|
|
|
name: string;
|
|
|
|
active: boolean;
|
|
|
|
configData: Record<string, any>;
|
|
|
|
token: string;
|
|
|
|
ast: any[];
|
2023-09-02 01:27:33 +02:00
|
|
|
}
|
2020-07-18 07:28:32 +02:00
|
|
|
|
2023-11-16 21:18:19 +01:00
|
|
|
import darkTheme from "@/themes/d-rosepine.json5";
|
2020-12-19 02:55:52 +01:00
|
|
|
/**
|
2023-10-17 03:57:20 +02:00
|
|
|
* Storage for configuration information that does not need to be constantly loaded into memory (non-reactive)
|
2020-12-19 02:55:52 +01:00
|
|
|
*/
|
2023-01-13 05:40:33 +01:00
|
|
|
import lightTheme from "@/themes/l-rosepinedawn.json5";
|
2022-05-01 15:51:07 +02:00
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
export class ColdDeviceStorage {
|
|
|
|
public static default = {
|
2022-05-01 15:51:07 +02:00
|
|
|
lightTheme,
|
|
|
|
darkTheme,
|
2020-12-19 02:55:52 +01:00
|
|
|
syncDeviceDarkMode: true,
|
|
|
|
plugins: [] as Plugin[],
|
|
|
|
mediaVolume: 0.5,
|
2024-03-01 00:51:19 +01:00
|
|
|
vibrate: false,
|
2020-12-19 02:55:52 +01:00
|
|
|
sound_masterVolume: 0.3,
|
2023-03-29 22:38:59 +02:00
|
|
|
sound_note: { type: "none", volume: 0 },
|
2023-01-13 05:40:33 +01:00
|
|
|
sound_noteMy: { type: "syuilo/up", volume: 1 },
|
|
|
|
sound_notification: { type: "syuilo/pope2", volume: 1 },
|
|
|
|
sound_chat: { type: "syuilo/pope1", volume: 1 },
|
|
|
|
sound_chatBg: { type: "syuilo/waon", volume: 1 },
|
|
|
|
sound_antenna: { type: "syuilo/triple", volume: 1 },
|
|
|
|
sound_channel: { type: "syuilo/square-pico", volume: 1 },
|
2020-12-19 02:55:52 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
public static watchers = [];
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
public static get<T extends keyof typeof ColdDeviceStorage.default>(
|
|
|
|
key: T,
|
2024-02-11 18:50:57 +01:00
|
|
|
): (typeof ColdDeviceStorage.default)[T] {
|
2020-12-19 02:55:52 +01:00
|
|
|
// TODO: indexedDBにする
|
|
|
|
// ただしその際はnullチェックではなくキー存在チェックにしないとダメ
|
|
|
|
// (indexedDBはnullを保存できるため、ユーザーが意図してnullを格納した可能性がある)
|
|
|
|
const value = localStorage.getItem(PREFIX + key);
|
|
|
|
if (value == null) {
|
|
|
|
return ColdDeviceStorage.default[key];
|
|
|
|
} else {
|
|
|
|
return JSON.parse(value);
|
|
|
|
}
|
|
|
|
}
|
2020-07-28 12:02:28 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
public static set<T extends keyof typeof ColdDeviceStorage.default>(
|
|
|
|
key: T,
|
2024-02-11 18:50:57 +01:00
|
|
|
value: (typeof ColdDeviceStorage.default)[T],
|
2023-01-13 05:40:33 +01:00
|
|
|
): void {
|
2022-07-06 00:08:45 +02:00
|
|
|
// 呼び出し側のバグ等で undefined が来ることがある
|
|
|
|
// undefined を文字列として localStorage に入れると参照する際の JSON.parse でコケて不具合の元になるため無視
|
|
|
|
if (value === undefined) {
|
|
|
|
console.error(`attempt to store undefined value for key '${key}'`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
localStorage.setItem(PREFIX + key, JSON.stringify(value));
|
2020-01-29 20:37:25 +01:00
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
for (const watcher of this.watchers) {
|
|
|
|
if (watcher.key === key) watcher.callback(value);
|
|
|
|
}
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
public static watch(key, callback) {
|
|
|
|
this.watchers.push({ key, callback });
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
// TODO: VueのcustomRef使うと良い感じになるかも
|
|
|
|
public static ref<T extends keyof typeof ColdDeviceStorage.default>(key: T) {
|
|
|
|
const v = ColdDeviceStorage.get(key);
|
|
|
|
const r = ref(v);
|
|
|
|
// TODO: このままではwatcherがリークするので開放する方法を考える
|
2023-01-13 05:40:33 +01:00
|
|
|
this.watch(key, (v) => {
|
2020-12-19 02:55:52 +01:00
|
|
|
r.value = v;
|
|
|
|
});
|
|
|
|
return r;
|
|
|
|
}
|
2020-01-29 20:37:25 +01:00
|
|
|
|
2020-12-19 02:55:52 +01:00
|
|
|
/**
|
|
|
|
* 特定のキーの、簡易的なgetter/setterを作ります
|
|
|
|
* 主にvue場で設定コントロールのmodelとして使う用
|
|
|
|
*/
|
2023-01-13 05:40:33 +01:00
|
|
|
public static makeGetterSetter<
|
|
|
|
K extends keyof typeof ColdDeviceStorage.default,
|
|
|
|
>(key: K) {
|
2020-12-19 02:55:52 +01:00
|
|
|
// TODO: VueのcustomRef使うと良い感じになるかも
|
|
|
|
const valueRef = ColdDeviceStorage.ref(key);
|
|
|
|
return {
|
|
|
|
get: () => {
|
|
|
|
return valueRef.value;
|
2020-01-29 20:37:25 +01:00
|
|
|
},
|
2020-12-19 02:55:52 +01:00
|
|
|
set: (value: unknown) => {
|
|
|
|
const val = value;
|
|
|
|
ColdDeviceStorage.set(key, val);
|
2022-06-27 17:27:24 +02:00
|
|
|
},
|
2020-12-19 02:55:52 +01:00
|
|
|
};
|
2020-01-29 20:37:25 +01:00
|
|
|
}
|
2020-12-19 02:55:52 +01:00
|
|
|
}
|