2026-05-28 19:28:52 -04:00
|
|
|
import { useEffect, useRef } from 'react';
|
|
|
|
|
import { useMatrixClient } from './useMatrixClient';
|
|
|
|
|
import { useSetting } from '../state/hooks/settings';
|
|
|
|
|
import { settingsAtom } from '../state/settings';
|
|
|
|
|
|
2026-05-30 23:29:29 -04:00
|
|
|
const IDLE_TIMEOUT_MS = 10 * 60 * 1000;
|
|
|
|
|
const ACTIVITY_THROTTLE_MS = 1000;
|
2026-05-28 19:28:52 -04:00
|
|
|
|
|
|
|
|
export function usePresenceUpdater() {
|
|
|
|
|
const mx = useMatrixClient();
|
|
|
|
|
const [hidePresence] = useSetting(settingsAtom, 'hidePresence');
|
2026-05-30 23:29:29 -04:00
|
|
|
const [presenceStatus] = useSetting(settingsAtom, 'presenceStatus');
|
2026-05-28 19:28:52 -04:00
|
|
|
|
|
|
|
|
const idleTimerRef = useRef<ReturnType<typeof setTimeout> | undefined>(undefined);
|
|
|
|
|
const isIdleRef = useRef(false);
|
|
|
|
|
const lastActivityRef = useRef(0);
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
2026-05-28 19:40:15 -04:00
|
|
|
const setOnline = () => mx.setPresence({ presence: 'online' }).catch(() => undefined);
|
2026-05-30 23:29:29 -04:00
|
|
|
const setUnavailable = (statusMsg?: string) =>
|
|
|
|
|
mx.setPresence({ presence: 'unavailable', status_msg: statusMsg }).catch(() => undefined);
|
|
|
|
|
const setOffline = () => mx.setPresence({ presence: 'offline' }).catch(() => undefined);
|
2026-05-28 19:28:52 -04:00
|
|
|
|
2026-05-30 23:29:29 -04:00
|
|
|
// Manual presence overrides — no activity tracking needed.
|
|
|
|
|
if (hidePresence || presenceStatus === 'invisible') {
|
|
|
|
|
setOffline();
|
|
|
|
|
return undefined;
|
|
|
|
|
}
|
|
|
|
|
if (presenceStatus === 'online') {
|
|
|
|
|
setOnline();
|
|
|
|
|
return undefined;
|
|
|
|
|
}
|
|
|
|
|
if (presenceStatus === 'idle') {
|
|
|
|
|
setUnavailable();
|
|
|
|
|
return undefined;
|
|
|
|
|
}
|
|
|
|
|
if (presenceStatus === 'dnd') {
|
|
|
|
|
setUnavailable('dnd');
|
2026-05-28 19:28:52 -04:00
|
|
|
return undefined;
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-30 23:29:29 -04:00
|
|
|
// presenceStatus === 'auto' — original activity-tracking behavior.
|
2026-05-28 19:28:52 -04:00
|
|
|
const startIdleTimer = () => {
|
|
|
|
|
clearTimeout(idleTimerRef.current);
|
|
|
|
|
idleTimerRef.current = window.setTimeout(() => {
|
|
|
|
|
isIdleRef.current = true;
|
|
|
|
|
setUnavailable();
|
|
|
|
|
}, IDLE_TIMEOUT_MS);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handleActivity = () => {
|
|
|
|
|
const now = Date.now();
|
|
|
|
|
if (now - lastActivityRef.current < ACTIVITY_THROTTLE_MS) return;
|
|
|
|
|
lastActivityRef.current = now;
|
|
|
|
|
|
|
|
|
|
if (isIdleRef.current && !document.hidden) {
|
|
|
|
|
isIdleRef.current = false;
|
|
|
|
|
setOnline();
|
|
|
|
|
}
|
|
|
|
|
startIdleTimer();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handleVisibilityChange = () => {
|
|
|
|
|
if (document.hidden) {
|
|
|
|
|
clearTimeout(idleTimerRef.current);
|
|
|
|
|
setUnavailable();
|
|
|
|
|
} else {
|
|
|
|
|
isIdleRef.current = false;
|
|
|
|
|
lastActivityRef.current = Date.now();
|
|
|
|
|
setOnline();
|
|
|
|
|
startIdleTimer();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handlePageHide = () => {
|
|
|
|
|
const userId = mx.getUserId();
|
|
|
|
|
const token = mx.getAccessToken();
|
|
|
|
|
const baseUrl = (mx as unknown as { baseUrl: string }).baseUrl;
|
|
|
|
|
if (!userId || !token || !baseUrl) return;
|
|
|
|
|
|
|
|
|
|
fetch(`${baseUrl}/_matrix/client/v3/presence/${encodeURIComponent(userId)}/status`, {
|
|
|
|
|
method: 'PUT',
|
|
|
|
|
headers: {
|
|
|
|
|
Authorization: `Bearer ${token}`,
|
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
|
},
|
|
|
|
|
body: JSON.stringify({ presence: 'offline' }),
|
|
|
|
|
keepalive: true,
|
|
|
|
|
}).catch(() => undefined);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
setOnline();
|
|
|
|
|
startIdleTimer();
|
|
|
|
|
|
|
|
|
|
const activityEvents = ['mousemove', 'keydown', 'touchstart', 'click', 'scroll'] as const;
|
|
|
|
|
activityEvents.forEach((e) => window.addEventListener(e, handleActivity, { passive: true }));
|
|
|
|
|
document.addEventListener('visibilitychange', handleVisibilityChange);
|
|
|
|
|
window.addEventListener('pagehide', handlePageHide);
|
|
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
|
clearTimeout(idleTimerRef.current);
|
|
|
|
|
activityEvents.forEach((e) => window.removeEventListener(e, handleActivity));
|
|
|
|
|
document.removeEventListener('visibilitychange', handleVisibilityChange);
|
|
|
|
|
window.removeEventListener('pagehide', handlePageHide);
|
|
|
|
|
};
|
2026-05-30 23:29:29 -04:00
|
|
|
}, [mx, hidePresence, presenceStatus]);
|
2026-05-28 19:28:52 -04:00
|
|
|
}
|