Files
cinny/src/app/hooks/useExtendedProfile.ts
T
jared b41bfd35c0
CI / Build & Quality Checks (push) Successful in 10m36s
Trigger Desktop Build / trigger (push) Successful in 12s
fix: persist status message and timezone across reconnects
Status message: Synapse clears status_msg when a user goes offline/reconnects.
Fix by caching to localStorage and re-sending on setOnline(). The sync
effect no longer overwrites the local value with an empty server event.

Timezone: PUT /profile/{userId}/m.tz is MSC1769 (unstable) and not
supported by standard Synapse — save/load silently fails. Fix by using
Matrix account data (im.lotus.timezone) instead, which is fully
supported. Own profile falls back to account data; other users still
try the m.tz profile endpoint (for federated servers that support it).

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-06-10 21:31:58 -04:00

66 lines
1.8 KiB
TypeScript

import { useEffect, useState } from 'react';
import { Method } from 'matrix-js-sdk';
import { useMatrixClient } from './useMatrixClient';
export type ExtendedProfile = {
pronouns?: string;
timezone?: string;
};
export const useExtendedProfile = (userId: string): ExtendedProfile => {
const mx = useMatrixClient();
const [extProfile, setExtProfile] = useState<ExtendedProfile>({});
useEffect(() => {
let cancelled = false;
const myUserId = mx.getUserId();
const fetchField = async <T extends Record<string, string>>(
field: string,
): Promise<string | undefined> => {
try {
const res = await mx.http.authedRequest<T>(
Method.Get,
`/profile/${encodeURIComponent(userId)}/${field}`,
);
return res[field as keyof T] as string | undefined;
} catch {
return undefined;
}
};
const run = async () => {
const [pronouns, tzFromProfile] = await Promise.all([
fetchField<{ 'm.pronouns': string }>('m.pronouns'),
fetchField<{ 'm.tz': string }>('m.tz'),
]);
let timezone = tzFromProfile;
// Standard Synapse doesn't support m.tz — fall back to account data for own profile
if (!timezone && userId === myUserId) {
try {
const res = await mx.http.authedRequest<{ timezone: string }>(
Method.Get,
`/user/${encodeURIComponent(userId)}/account_data/im.lotus.timezone`,
);
timezone = res.timezone || undefined;
} catch {
// not set yet
}
}
if (!cancelled) {
setExtProfile({ pronouns: pronouns || undefined, timezone: timezone || undefined });
}
};
run();
return () => {
cancelled = true;
};
}, [mx, userId]);
return extProfile;
};