2023-01-13 10:30:09 +03:00
|
|
|
import './drafts.css';
|
|
|
|
|
|
|
|
import { useEffect, useMemo, useReducer, useState } from 'react';
|
|
|
|
|
2023-02-05 19:17:19 +03:00
|
|
|
import { api } from '../utils/api';
|
2023-01-13 10:30:09 +03:00
|
|
|
import db from '../utils/db';
|
2023-03-01 15:07:22 +03:00
|
|
|
import niceDateTime from '../utils/nice-date-time';
|
2023-01-13 10:30:09 +03:00
|
|
|
import states from '../utils/states';
|
|
|
|
import { getCurrentAccountNS } from '../utils/store-utils';
|
|
|
|
|
|
|
|
import Icon from './icon';
|
|
|
|
import Loader from './loader';
|
2023-07-17 16:01:00 +03:00
|
|
|
import MenuConfirm from './menu-confirm';
|
2023-01-13 10:30:09 +03:00
|
|
|
|
2023-04-20 11:10:57 +03:00
|
|
|
function Drafts({ onClose }) {
|
2023-02-05 19:17:19 +03:00
|
|
|
const { masto } = api();
|
2023-01-13 10:30:09 +03:00
|
|
|
const [uiState, setUIState] = useState('default');
|
|
|
|
const [drafts, setDrafts] = useState([]);
|
|
|
|
const [reloadCount, reload] = useReducer((c) => c + 1, 0);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
setUIState('loading');
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
const keys = await db.drafts.keys();
|
|
|
|
if (keys.length) {
|
|
|
|
const ns = getCurrentAccountNS();
|
|
|
|
const ownKeys = keys.filter((key) => key.startsWith(ns));
|
|
|
|
if (ownKeys.length) {
|
|
|
|
const drafts = await db.drafts.getMany(ownKeys);
|
|
|
|
drafts.sort(
|
|
|
|
(a, b) =>
|
|
|
|
new Date(b.updatedAt).getTime() -
|
|
|
|
new Date(a.updatedAt).getTime(),
|
|
|
|
);
|
|
|
|
setDrafts(drafts);
|
|
|
|
} else {
|
|
|
|
setDrafts([]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
setDrafts([]);
|
|
|
|
}
|
|
|
|
setUIState('default');
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
setUIState('error');
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}, [reloadCount]);
|
|
|
|
|
|
|
|
const hasDrafts = drafts?.length > 0;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div class="sheet">
|
2023-04-20 11:10:57 +03:00
|
|
|
{!!onClose && (
|
|
|
|
<button type="button" class="sheet-close" onClick={onClose}>
|
|
|
|
<Icon icon="x" />
|
|
|
|
</button>
|
|
|
|
)}
|
2023-01-13 10:30:09 +03:00
|
|
|
<header>
|
|
|
|
<h2>
|
|
|
|
Unsent drafts <Loader abrupt hidden={uiState !== 'loading'} />
|
|
|
|
</h2>
|
|
|
|
{hasDrafts && (
|
|
|
|
<div class="insignificant">
|
|
|
|
Looks like you have unsent drafts. Let's continue where you left
|
|
|
|
off.
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</header>
|
|
|
|
<main>
|
|
|
|
{hasDrafts ? (
|
|
|
|
<>
|
|
|
|
<ul class="drafts-list">
|
|
|
|
{drafts.map((draft) => {
|
|
|
|
const { updatedAt, key, draftStatus, replyTo } = draft;
|
|
|
|
const updatedAtDate = new Date(updatedAt);
|
|
|
|
return (
|
|
|
|
<li key={updatedAt}>
|
|
|
|
<div class="mini-draft-meta">
|
|
|
|
<b>
|
|
|
|
<Icon icon={replyTo ? 'reply' : 'quill'} size="s" />{' '}
|
|
|
|
<time>
|
|
|
|
{!!replyTo && (
|
|
|
|
<>
|
|
|
|
@{replyTo.account.acct}
|
|
|
|
<br />
|
|
|
|
</>
|
|
|
|
)}
|
2023-03-01 15:07:22 +03:00
|
|
|
{niceDateTime(updatedAtDate)}
|
2023-01-13 10:30:09 +03:00
|
|
|
</time>
|
|
|
|
</b>
|
2023-07-17 16:01:00 +03:00
|
|
|
<MenuConfirm
|
|
|
|
confirmLabel={<span>Delete this draft?</span>}
|
|
|
|
menuItemClassName="danger"
|
|
|
|
align="end"
|
2023-01-13 10:30:09 +03:00
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
onClick={() => {
|
|
|
|
(async () => {
|
|
|
|
try {
|
2023-07-17 16:01:00 +03:00
|
|
|
// const yes = confirm('Delete this draft?');
|
|
|
|
// if (yes) {
|
|
|
|
await db.drafts.del(key);
|
|
|
|
reload();
|
|
|
|
// }
|
2023-01-13 10:30:09 +03:00
|
|
|
} catch (e) {
|
|
|
|
alert('Error deleting draft! Please try again.');
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}}
|
|
|
|
>
|
2023-07-17 16:01:00 +03:00
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="small light"
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
>
|
|
|
|
Delete…
|
|
|
|
</button>
|
|
|
|
</MenuConfirm>
|
2023-01-13 10:30:09 +03:00
|
|
|
</div>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
class="draft-item"
|
|
|
|
onClick={async () => {
|
|
|
|
// console.log({ draftStatus });
|
|
|
|
let replyToStatus;
|
|
|
|
if (replyTo) {
|
|
|
|
setUIState('loading');
|
|
|
|
try {
|
|
|
|
replyToStatus = await masto.v1.statuses.fetch(
|
|
|
|
replyTo.id,
|
|
|
|
);
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
alert('Error fetching reply-to status!');
|
|
|
|
setUIState('default');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setUIState('default');
|
|
|
|
}
|
|
|
|
window.__COMPOSE__ = {
|
|
|
|
draftStatus,
|
|
|
|
replyToStatus,
|
|
|
|
};
|
|
|
|
states.showCompose = true;
|
|
|
|
states.showDrafts = false;
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<MiniDraft draft={draft} />
|
|
|
|
</button>
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</ul>
|
2023-07-17 16:01:00 +03:00
|
|
|
{drafts.length > 1 && (
|
|
|
|
<p>
|
|
|
|
<MenuConfirm
|
|
|
|
confirmLabel={<span>Delete all drafts?</span>}
|
|
|
|
menuItemClassName="danger"
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
onClick={() => {
|
|
|
|
(async () => {
|
|
|
|
// const yes = confirm('Delete all drafts?');
|
|
|
|
// if (yes) {
|
2023-01-13 10:30:09 +03:00
|
|
|
setUIState('loading');
|
|
|
|
try {
|
|
|
|
await db.drafts.delMany(
|
|
|
|
drafts.map((draft) => draft.key),
|
|
|
|
);
|
|
|
|
setUIState('default');
|
|
|
|
reload();
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
alert('Error deleting drafts! Please try again.');
|
|
|
|
setUIState('error');
|
|
|
|
}
|
2023-07-17 16:01:00 +03:00
|
|
|
// }
|
|
|
|
})();
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="light danger"
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
>
|
|
|
|
Delete all…
|
|
|
|
</button>
|
|
|
|
</MenuConfirm>
|
|
|
|
</p>
|
|
|
|
)}
|
2023-01-13 10:30:09 +03:00
|
|
|
</>
|
|
|
|
) : (
|
|
|
|
<p>No drafts found.</p>
|
|
|
|
)}
|
|
|
|
</main>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function MiniDraft({ draft }) {
|
|
|
|
const { draftStatus, replyTo } = draft;
|
|
|
|
const { status, spoilerText, poll, mediaAttachments } = draftStatus;
|
|
|
|
const hasPoll = poll?.options?.length > 0;
|
|
|
|
const hasMedia = mediaAttachments?.length > 0;
|
|
|
|
const hasPollOrMedia = hasPoll || hasMedia;
|
|
|
|
const firstImageMedia = useMemo(() => {
|
|
|
|
if (!hasMedia) return;
|
|
|
|
const image = mediaAttachments.find((media) => /image/.test(media.type));
|
|
|
|
if (!image) return;
|
|
|
|
const { file } = image;
|
|
|
|
const objectURL = URL.createObjectURL(file);
|
|
|
|
return objectURL;
|
|
|
|
}, [hasMedia, mediaAttachments]);
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<div class="mini-draft">
|
|
|
|
{hasPollOrMedia && (
|
|
|
|
<div
|
|
|
|
class={`mini-draft-aside ${firstImageMedia ? 'has-image' : ''}`}
|
|
|
|
style={
|
|
|
|
firstImageMedia
|
|
|
|
? {
|
|
|
|
'--bg-image': `url(${firstImageMedia})`,
|
|
|
|
}
|
|
|
|
: {}
|
|
|
|
}
|
|
|
|
>
|
|
|
|
{hasPoll && <Icon icon="poll" />}
|
|
|
|
{hasMedia && (
|
|
|
|
<span>
|
|
|
|
<Icon icon="attachment" />{' '}
|
|
|
|
<small>{mediaAttachments?.length}</small>
|
|
|
|
</span>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<div class="mini-draft-main">
|
|
|
|
{!!spoilerText && <div class="mini-draft-spoiler">{spoilerText}</div>}
|
|
|
|
{!!status && <div class="mini-draft-status">{status}</div>}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Drafts;
|