1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
|
"use client";
import {
createContext,
ReactNode,
useCallback,
useContext,
useEffect,
useMemo,
useState,
} from "react";
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query";
import {
READER_DEFAULTS,
ReaderSettings,
ReaderSettingsPartial,
} from "@karakeep/shared/types/readers";
import { useTRPC } from "../trpc";
export interface UseReaderSettingsOptions {
/**
* Get local overrides (device-specific settings stored locally)
*/
getLocalOverrides: () => ReaderSettingsPartial;
/**
* Save local overrides to local storage
*/
saveLocalOverrides: (overrides: ReaderSettingsPartial) => void;
/**
* Optional session overrides (for live preview in web).
* If provided, these take highest precedence.
*/
sessionOverrides?: ReaderSettingsPartial;
/**
* Callback when session overrides should be cleared (after successful server save)
*/
onClearSessionOverrides?: () => void;
}
export function useReaderSettings(options: UseReaderSettingsOptions) {
const api = useTRPC();
const {
getLocalOverrides,
saveLocalOverrides,
sessionOverrides = {},
onClearSessionOverrides,
} = options;
const [localOverrides, setLocalOverrides] = useState<ReaderSettingsPartial>(
{},
);
const [pendingServerSave, setPendingServerSave] =
useState<ReaderSettings | null>(null);
const { data: serverSettings } = useQuery(api.users.settings.queryOptions());
const queryClient = useQueryClient();
// Load local overrides on mount
useEffect(() => {
setLocalOverrides(getLocalOverrides());
}, [getLocalOverrides]);
// Clear pending state when server settings match what we saved
useEffect(() => {
if (pendingServerSave && serverSettings) {
const serverMatches =
serverSettings.readerFontSize === pendingServerSave.fontSize &&
// Tolerate minor float normalization differences for lineHeight
Math.abs(
(serverSettings.readerLineHeight ?? 0) - pendingServerSave.lineHeight,
) < 1e-6 &&
serverSettings.readerFontFamily === pendingServerSave.fontFamily;
if (serverMatches) {
setPendingServerSave(null);
}
}
}, [serverSettings, pendingServerSave]);
const { mutate: updateServerSettings, isPending: isSaving } = useMutation(
api.users.updateSettings.mutationOptions({
onSettled: async () => {
await queryClient.refetchQueries(api.users.settings.pathFilter());
},
}),
);
// Separate mutation for saving defaults (clears local overrides on success)
const { mutate: saveServerSettings, isPending: isSavingDefaults } =
useMutation(
api.users.updateSettings.mutationOptions({
onSuccess: () => {
// Clear local and session overrides after successful server save
setLocalOverrides({});
saveLocalOverrides({});
onClearSessionOverrides?.();
},
onError: () => {
// Clear pending state so we don't show values that failed to persist
setPendingServerSave(null);
},
onSettled: async () => {
await queryClient.refetchQueries(api.users.settings.pathFilter());
},
}),
);
// Compute effective settings with precedence: session → local → pendingSave → server → default
const settings: ReaderSettings = useMemo(
() => ({
fontSize:
sessionOverrides.fontSize ??
localOverrides.fontSize ??
pendingServerSave?.fontSize ??
serverSettings?.readerFontSize ??
READER_DEFAULTS.fontSize,
lineHeight:
sessionOverrides.lineHeight ??
localOverrides.lineHeight ??
pendingServerSave?.lineHeight ??
serverSettings?.readerLineHeight ??
READER_DEFAULTS.lineHeight,
fontFamily:
sessionOverrides.fontFamily ??
localOverrides.fontFamily ??
pendingServerSave?.fontFamily ??
serverSettings?.readerFontFamily ??
READER_DEFAULTS.fontFamily,
}),
[sessionOverrides, localOverrides, pendingServerSave, serverSettings],
);
// Get the server setting values (for UI indicators)
const serverDefaults: ReaderSettingsPartial = useMemo(
() => ({
fontSize: serverSettings?.readerFontSize ?? undefined,
lineHeight: serverSettings?.readerLineHeight ?? undefined,
fontFamily: serverSettings?.readerFontFamily ?? undefined,
}),
[serverSettings],
);
// Update local override (per-device, immediate)
const updateLocal = useCallback(
(updates: ReaderSettingsPartial) => {
setLocalOverrides((prev) => {
const newOverrides = { ...prev, ...updates };
saveLocalOverrides(newOverrides);
return newOverrides;
});
},
[saveLocalOverrides],
);
// Clear a specific local override
const clearLocal = useCallback(
(key: keyof ReaderSettings) => {
setLocalOverrides((prev) => {
const { [key]: _, ...rest } = prev;
saveLocalOverrides(rest);
return rest;
});
},
[saveLocalOverrides],
);
// Clear all local overrides
const clearAllLocal = useCallback(() => {
setLocalOverrides({});
saveLocalOverrides({});
}, [saveLocalOverrides]);
// Save current effective settings as server default (syncs across devices)
const saveAsDefault = useCallback(
(settingsToSave?: ReaderSettingsPartial) => {
const toSave: ReaderSettings = {
fontSize: settingsToSave?.fontSize ?? settings.fontSize,
lineHeight: settingsToSave?.lineHeight ?? settings.lineHeight,
fontFamily: settingsToSave?.fontFamily ?? settings.fontFamily,
};
// Set pending state to prevent flicker while server syncs
setPendingServerSave(toSave);
saveServerSettings({
readerFontSize: toSave.fontSize,
readerLineHeight: toSave.lineHeight,
readerFontFamily: toSave.fontFamily,
});
},
[settings, saveServerSettings],
);
// Clear a specific server default (set to null)
const clearDefault = useCallback(
(key: keyof ReaderSettings) => {
const serverKeyMap = {
fontSize: "readerFontSize",
lineHeight: "readerLineHeight",
fontFamily: "readerFontFamily",
} as const;
updateServerSettings({ [serverKeyMap[key]]: null });
},
[updateServerSettings],
);
// Clear all server defaults
const clearAllDefaults = useCallback(() => {
updateServerSettings({
readerFontSize: null,
readerLineHeight: null,
readerFontFamily: null,
});
}, [updateServerSettings]);
// Check if there are any local overrides
const hasLocalOverrides = Object.keys(localOverrides).length > 0;
// Check if there are any server defaults
const hasServerDefaults =
serverSettings?.readerFontSize != null ||
serverSettings?.readerLineHeight != null ||
serverSettings?.readerFontFamily != null;
return {
// Current effective settings (what should be displayed)
settings,
// Raw values for UI indicators
localOverrides,
serverDefaults,
// Status flags
hasLocalOverrides,
hasServerDefaults,
isSaving: isSaving || isSavingDefaults,
// Internal state setters (for web's context-based approach)
setLocalOverrides,
// Actions
updateLocal,
clearLocal,
clearAllLocal,
saveAsDefault,
clearDefault,
clearAllDefaults,
};
}
// Context for sharing reader settings state across components
export type ReaderSettingsContextValue = ReturnType<typeof useReaderSettings>;
const ReaderSettingsContext = createContext<ReaderSettingsContextValue | null>(
null,
);
export interface ReaderSettingsProviderProps extends UseReaderSettingsOptions {
children: ReactNode;
}
/**
* Provider that creates a single instance of reader settings state
* and shares it across all child components.
*/
export function ReaderSettingsProvider({
children,
...options
}: ReaderSettingsProviderProps) {
const value = useReaderSettings(options);
return (
<ReaderSettingsContext.Provider value={value}>
{children}
</ReaderSettingsContext.Provider>
);
}
/**
* Hook to access shared reader settings from context.
* Must be used within a ReaderSettingsProvider.
*/
export function useReaderSettingsContext() {
const context = useContext(ReaderSettingsContext);
if (!context) {
throw new Error(
"useReaderSettingsContext must be used within a ReaderSettingsProvider",
);
}
return context;
}
|