Files
fx-test/fluxer_app/src/stores/LocalPresenceStore.tsx
2026-01-03 06:44:40 +01:00

104 lines
2.9 KiB
TypeScript

/*
* Copyright (C) 2026 Fluxer Contributors
*
* This file is part of Fluxer.
*
* Fluxer is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Fluxer is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Fluxer. If not, see <https://www.gnu.org/licenses/>.
*/
import {makeAutoObservable, reaction} from 'mobx';
import type {StatusType} from '~/Constants';
import {StatusTypes} from '~/Constants';
import {
type CustomStatus,
customStatusToKey,
type GatewayCustomStatusPayload,
normalizeCustomStatus,
toGatewayCustomStatus,
} from '~/lib/customStatus';
import IdleStore from '~/stores/IdleStore';
import MobileLayoutStore from '~/stores/MobileLayoutStore';
import UserSettingsStore from '~/stores/UserSettingsStore';
type Presence = Readonly<{
status: StatusType;
since: number;
afk: boolean;
mobile: boolean;
custom_status: GatewayCustomStatusPayload | null;
}>;
class LocalPresenceStore {
status: StatusType = StatusTypes.ONLINE;
since: number = 0;
afk: boolean = false;
mobile: boolean = false;
customStatus: CustomStatus | null = null;
constructor() {
makeAutoObservable(this, {}, {autoBind: true});
reaction(
() => MobileLayoutStore.isMobileLayout(),
() => this.updatePresence(),
);
}
updatePresence(): void {
const userStatus = UserSettingsStore.status;
const idleSince = IdleStore.getIdleSince();
const isMobile = MobileLayoutStore.isMobileLayout();
const afk = this.computeAfk(idleSince, isMobile);
const effectiveStatus = userStatus === StatusTypes.ONLINE && idleSince > 0 ? StatusTypes.IDLE : userStatus;
const normalizedCustomStatus = normalizeCustomStatus(UserSettingsStore.getCustomStatus());
this.customStatus = normalizedCustomStatus ? {...normalizedCustomStatus} : null;
this.status = effectiveStatus;
this.since = idleSince;
this.afk = afk;
this.mobile = isMobile;
}
getStatus(): StatusType {
return this.status;
}
getPresence(): Presence {
return {
status: this.status,
since: this.since,
afk: this.afk,
mobile: this.mobile,
custom_status: toGatewayCustomStatus(this.customStatus),
};
}
get presenceFingerprint(): string {
return `${this.status}|${customStatusToKey(this.customStatus)}|afk:${this.afk ? '1' : '0'}`;
}
private computeAfk(idleSince: number, isMobile: boolean): boolean {
if (isMobile || idleSince <= 0) return false;
const afkTimeout = UserSettingsStore.getAfkTimeout();
return Date.now() - idleSince > afkTimeout * 1000;
}
}
export default new LocalPresenceStore();