127 lines
3.9 KiB
TypeScript
Raw Normal View History

import { useEffect, useState } from 'react';
import {
fluentPadServiceId,
notifyOnlineFnName,
notifyUserAddedFnName,
notifyUserRemovedFnName,
2021-01-16 13:24:26 +03:00
} from 'src/app/constants';
import { useFluenceClient } from '../app/FluenceClientContext';
2021-01-16 16:28:00 +03:00
import * as api from 'src/app/api';
2021-01-14 21:52:33 +03:00
import { subscribeToEvent } from '@fluencelabs/fluence';
2021-01-16 16:25:28 +03:00
type PeerId = string;
interface User {
2021-01-16 16:25:28 +03:00
id: PeerId;
name: string;
isOnline: boolean;
shouldBecomeOnline: boolean;
}
const turnUserAsOfflineCandidate = (u: User): User => {
return {
...u,
isOnline: u.shouldBecomeOnline,
shouldBecomeOnline: false,
};
};
2021-01-14 21:52:33 +03:00
const refreshTimeoutMs = 2000;
export const UserList = (props: { selfName: string }) => {
const client = useFluenceClient()!;
const [users, setUsers] = useState<Map<PeerId, User>>(new Map());
useEffect(() => {
const listRefreshTimer = setInterval(() => {
setUsers((prev) => {
const newUsers = Array.from(prev).map(
([key, user]) => [key, turnUserAsOfflineCandidate(user)] as const,
);
return new Map(newUsers);
});
// don't block
2021-01-16 16:28:00 +03:00
api.updateOnlineStatuses(client);
}, refreshTimeoutMs);
const unsub1 = subscribeToEvent(client, fluentPadServiceId, notifyUserAddedFnName, (args, _) => {
2021-01-16 16:28:00 +03:00
const users = args.flatMap((x) => x).flatMap((x) => x) as api.User[];
setUsers((prev) => {
const result = new Map(prev);
for (let u of users) {
if (result.has(u.peer_id)) {
continue;
}
const isCurrentUser = u.peer_id === client.selfPeerId.toB58String();
result.set(u.peer_id, {
name: u.name,
id: u.peer_id,
isOnline: isCurrentUser,
shouldBecomeOnline: isCurrentUser,
});
}
return result;
});
});
const unsub2 = subscribeToEvent(client, fluentPadServiceId, notifyUserRemovedFnName, (args, _) => {
const users = args.flatMap((x) => x) as PeerId[];
setUsers((prev) => {
const result = new Map(prev);
for (let u of users) {
result.delete(u);
}
return result;
});
});
const unsub3 = subscribeToEvent(client, fluentPadServiceId, notifyOnlineFnName, (args, _) => {
const [[peerId], immediately] = args;
setUsers((prev) => {
const result = new Map(prev);
const toSetOnline = result.get(peerId);
if (toSetOnline) {
toSetOnline.shouldBecomeOnline = true;
if (immediately) {
toSetOnline.isOnline = true;
}
}
return result;
});
});
// don't block
2021-01-16 16:28:00 +03:00
api.getUserList(client);
api.notifySelfAdded(client, props.selfName);
return () => {
clearTimeout(listRefreshTimer);
unsub1();
unsub2();
unsub3();
};
}, []);
const usersArray = Array.from(users)
.map((x) => x[1])
.sort((a, b) => a.name.localeCompare(b.name));
return (
2021-01-16 16:25:28 +03:00
<div className="userlist">
<ul>
2021-01-14 21:52:33 +03:00
{usersArray.map((x) => (
<li key={x.id}>
2021-01-16 16:25:28 +03:00
<span className={x.id === client.selfPeerId.toB58String() ? 'bold' : ''}>{x.name}</span>
<span className={x.isOnline ? 'green' : 'red'}> ({x.isOnline ? 'online' : 'offline'})</span>
</li>
))}
</ul>
</div>
);
};