Fix race conditions when accept/rejecting many follow requests
- No longer reload the whole list of follow requests and notifications for every accept/reject action - Notifications list now exclude follow requests (experimental)
This commit is contained in:
parent
37ce48ae6e
commit
8b74a32168
5 changed files with 52 additions and 12 deletions
|
@ -2,11 +2,17 @@ import { useState } from 'preact/hooks';
|
||||||
|
|
||||||
import { api } from '../utils/api';
|
import { api } from '../utils/api';
|
||||||
|
|
||||||
|
import Icon from './icon';
|
||||||
import Loader from './loader';
|
import Loader from './loader';
|
||||||
|
|
||||||
function FollowRequestButtons({ accountID, onChange }) {
|
function FollowRequestButtons({ accountID, onChange }) {
|
||||||
const { masto } = api();
|
const { masto } = api();
|
||||||
const [uiState, setUIState] = useState('default');
|
const [uiState, setUIState] = useState('default');
|
||||||
|
const [requestState, setRequestState] = useState(null); // accept, reject
|
||||||
|
const [relationship, setRelationship] = useState(null);
|
||||||
|
|
||||||
|
const hasRelationship = relationship !== null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<p class="follow-request-buttons">
|
<p class="follow-request-buttons">
|
||||||
<button
|
<button
|
||||||
|
@ -14,14 +20,19 @@ function FollowRequestButtons({ accountID, onChange }) {
|
||||||
disabled={uiState === 'loading'}
|
disabled={uiState === 'loading'}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setUIState('loading');
|
setUIState('loading');
|
||||||
|
setRequestState('accept');
|
||||||
(async () => {
|
(async () => {
|
||||||
try {
|
try {
|
||||||
await masto.v1.followRequests.authorize(accountID);
|
const rel = await masto.v1.followRequests.authorize(accountID);
|
||||||
|
if (!rel?.followedBy) {
|
||||||
|
throw new Error('Follow request not accepted');
|
||||||
|
}
|
||||||
|
setRelationship(rel);
|
||||||
onChange();
|
onChange();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error(e);
|
console.error(e);
|
||||||
setUIState('default');
|
|
||||||
}
|
}
|
||||||
|
setUIState('default');
|
||||||
})();
|
})();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
@ -33,9 +44,14 @@ function FollowRequestButtons({ accountID, onChange }) {
|
||||||
class="light danger"
|
class="light danger"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setUIState('loading');
|
setUIState('loading');
|
||||||
|
setRequestState('reject');
|
||||||
(async () => {
|
(async () => {
|
||||||
try {
|
try {
|
||||||
await masto.v1.followRequests.reject(accountID);
|
const rel = await masto.v1.followRequests.reject(accountID);
|
||||||
|
if (rel?.followedBy) {
|
||||||
|
throw new Error('Follow request not rejected');
|
||||||
|
}
|
||||||
|
setRelationship(rel);
|
||||||
onChange();
|
onChange();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error(e);
|
console.error(e);
|
||||||
|
@ -46,7 +62,17 @@ function FollowRequestButtons({ accountID, onChange }) {
|
||||||
>
|
>
|
||||||
Reject
|
Reject
|
||||||
</button>
|
</button>
|
||||||
<Loader hidden={uiState !== 'loading'} />
|
<span class="follow-request-states">
|
||||||
|
{hasRelationship && requestState ? (
|
||||||
|
requestState === 'accept' ? (
|
||||||
|
<Icon icon="check-circle" alt="Accepted" class="follow-accepted" />
|
||||||
|
) : (
|
||||||
|
<Icon icon="x-circle" alt="Rejected" class="follow-rejected" />
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
<Loader hidden={uiState !== 'loading'} />
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
</p>
|
</p>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,6 +13,7 @@ const ICONS = {
|
||||||
heart: () => import('@iconify-icons/mingcute/heart-line'),
|
heart: () => import('@iconify-icons/mingcute/heart-line'),
|
||||||
bookmark: () => import('@iconify-icons/mingcute/bookmark-line'),
|
bookmark: () => import('@iconify-icons/mingcute/bookmark-line'),
|
||||||
'check-circle': () => import('@iconify-icons/mingcute/check-circle-line'),
|
'check-circle': () => import('@iconify-icons/mingcute/check-circle-line'),
|
||||||
|
'x-circle': () => import('@iconify-icons/mingcute/close-circle-line'),
|
||||||
transfer: () => import('@iconify-icons/mingcute/transfer-4-line'),
|
transfer: () => import('@iconify-icons/mingcute/transfer-4-line'),
|
||||||
rocket: () => import('@iconify-icons/mingcute/rocket-line'),
|
rocket: () => import('@iconify-icons/mingcute/rocket-line'),
|
||||||
'arrow-left': () => import('@iconify-icons/mingcute/arrow-left-line'),
|
'arrow-left': () => import('@iconify-icons/mingcute/arrow-left-line'),
|
||||||
|
|
|
@ -137,7 +137,7 @@ function Notification({ notification, instance, reload }) {
|
||||||
<FollowRequestButtons
|
<FollowRequestButtons
|
||||||
accountID={account.id}
|
accountID={account.id}
|
||||||
onChange={() => {
|
onChange={() => {
|
||||||
reload();
|
// reload();
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
|
@ -204,7 +204,19 @@
|
||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
.follow-requests ul li .follow-request-buttons .loader-container {
|
.follow-request-buttons .follow-request-states {
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
.follow-request-buttons .follow-request-states .icon {
|
||||||
|
margin-inline: 8px;
|
||||||
|
}
|
||||||
|
.follow-request-buttons .follow-request-states .icon.follow-accepted {
|
||||||
|
color: var(--green-color);
|
||||||
|
}
|
||||||
|
.follow-request-buttons .follow-request-states .icon.follow-rejected {
|
||||||
|
color: var(--red-color);
|
||||||
|
}
|
||||||
|
.follow-requests ul li .follow-request-buttons .follow-request-states {
|
||||||
order: -1;
|
order: -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -49,6 +49,7 @@ function Notifications() {
|
||||||
// Reset iterator
|
// Reset iterator
|
||||||
notificationsIterator.current = masto.v1.notifications.list({
|
notificationsIterator.current = masto.v1.notifications.list({
|
||||||
limit: LIMIT,
|
limit: LIMIT,
|
||||||
|
excludeTypes: ['follow_request'],
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
const allNotifications = await notificationsIterator.current.next();
|
const allNotifications = await notificationsIterator.current.next();
|
||||||
|
@ -295,13 +296,13 @@ function Notifications() {
|
||||||
<summary>{followRequests.length} follow requests</summary>
|
<summary>{followRequests.length} follow requests</summary>
|
||||||
<ul>
|
<ul>
|
||||||
{followRequests.map((account) => (
|
{followRequests.map((account) => (
|
||||||
<li>
|
<li key={account.id}>
|
||||||
<AccountBlock account={account} />
|
<AccountBlock account={account} />
|
||||||
<FollowRequestButtons
|
<FollowRequestButtons
|
||||||
accountID={account.id}
|
accountID={account.id}
|
||||||
onChange={() => {
|
onChange={() => {
|
||||||
loadFollowRequests();
|
// loadFollowRequests();
|
||||||
loadNotifications(true);
|
// loadNotifications(true);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</li>
|
</li>
|
||||||
|
@ -311,13 +312,13 @@ function Notifications() {
|
||||||
) : (
|
) : (
|
||||||
<ul>
|
<ul>
|
||||||
{followRequests.map((account) => (
|
{followRequests.map((account) => (
|
||||||
<li>
|
<li key={account.id}>
|
||||||
<AccountBlock account={account} />
|
<AccountBlock account={account} />
|
||||||
<FollowRequestButtons
|
<FollowRequestButtons
|
||||||
accountID={account.id}
|
accountID={account.id}
|
||||||
onChange={() => {
|
onChange={() => {
|
||||||
loadFollowRequests();
|
// loadFollowRequests();
|
||||||
loadNotifications(true);
|
// loadNotifications(true);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</li>
|
</li>
|
||||||
|
|
Loading…
Add table
Reference in a new issue