2024-03-23 01:07:24 +08:00
|
|
|
import './filters.css';
|
|
|
|
|
2024-08-13 15:26:23 +08:00
|
|
|
import { i18n } from '@lingui/core';
|
|
|
|
import { msg, Plural, t, Trans } from '@lingui/macro';
|
|
|
|
import { useLingui } from '@lingui/react';
|
2024-03-23 01:07:24 +08:00
|
|
|
import { useEffect, useReducer, useRef, useState } from 'preact/hooks';
|
|
|
|
|
|
|
|
import Icon from '../components/icon';
|
|
|
|
import Link from '../components/link';
|
|
|
|
import Loader from '../components/loader';
|
|
|
|
import MenuConfirm from '../components/menu-confirm';
|
|
|
|
import Modal from '../components/modal';
|
|
|
|
import NavMenu from '../components/nav-menu';
|
|
|
|
import RelativeTime from '../components/relative-time';
|
|
|
|
import { api } from '../utils/api';
|
2024-08-13 15:26:23 +08:00
|
|
|
import i18nDuration from '../utils/i18n-duration';
|
2024-03-23 01:07:24 +08:00
|
|
|
import useInterval from '../utils/useInterval';
|
|
|
|
import useTitle from '../utils/useTitle';
|
|
|
|
|
|
|
|
const FILTER_CONTEXT = ['home', 'public', 'notifications', 'thread', 'account'];
|
|
|
|
const FILTER_CONTEXT_UNIMPLEMENTED = ['notifications', 'thread', 'account'];
|
|
|
|
const FILTER_CONTEXT_LABELS = {
|
2024-08-13 15:26:23 +08:00
|
|
|
home: msg`Home and lists`,
|
|
|
|
notifications: msg`Notifications`,
|
|
|
|
public: msg`Public timelines`,
|
|
|
|
thread: msg`Conversations`,
|
|
|
|
account: msg`Profiles`,
|
2024-03-23 01:07:24 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
const EXPIRY_DURATIONS = [
|
|
|
|
0, // forever
|
|
|
|
30 * 60, // 30 minutes
|
|
|
|
60 * 60, // 1 hour
|
|
|
|
6 * 60 * 60, // 6 hours
|
|
|
|
12 * 60 * 60, // 12 hours
|
|
|
|
60 * 60 * 24, // 24 hours
|
|
|
|
60 * 60 * 24 * 7, // 7 days
|
|
|
|
60 * 60 * 24 * 30, // 30 days
|
|
|
|
];
|
2024-08-13 15:26:23 +08:00
|
|
|
|
2024-03-23 01:07:24 +08:00
|
|
|
const EXPIRY_DURATIONS_LABELS = {
|
2024-08-13 15:26:23 +08:00
|
|
|
0: msg`Never`,
|
|
|
|
1800: i18nDuration(30, 'minute'),
|
|
|
|
3600: i18nDuration(1, 'hour'),
|
|
|
|
21600: i18nDuration(6, 'hour'),
|
|
|
|
43200: i18nDuration(12, 'hour'),
|
|
|
|
86_400: i18nDuration(24, 'hour'),
|
|
|
|
604_800: i18nDuration(7, 'day'),
|
|
|
|
2_592_000: i18nDuration(30, 'day'),
|
2024-03-23 01:07:24 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
function Filters() {
|
|
|
|
const { masto } = api();
|
2024-08-13 15:26:23 +08:00
|
|
|
useTitle(t`Filters`, `/ft`);
|
2024-03-23 01:07:24 +08:00
|
|
|
const [uiState, setUIState] = useState('default');
|
|
|
|
const [showFiltersAddEditModal, setShowFiltersAddEditModal] = useState(false);
|
|
|
|
|
|
|
|
const [reloadCount, reload] = useReducer((c) => c + 1, 0);
|
|
|
|
const [filters, setFilters] = useState([]);
|
|
|
|
useEffect(() => {
|
|
|
|
setUIState('loading');
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
const filters = await masto.v2.filters.list();
|
|
|
|
filters.sort((a, b) => a.title.localeCompare(b.title));
|
|
|
|
filters.forEach((filter) => {
|
|
|
|
if (filter.keywords?.length) {
|
|
|
|
filter.keywords.sort((a, b) => a.id - b.id);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
console.log(filters);
|
|
|
|
setFilters(filters);
|
|
|
|
setUIState('default');
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
setUIState('error');
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}, [reloadCount]);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div id="filters-page" class="deck-container" tabIndex="-1">
|
|
|
|
<div class="timeline-deck deck">
|
|
|
|
<header>
|
|
|
|
<div class="header-grid">
|
|
|
|
<div class="header-side">
|
|
|
|
<NavMenu />
|
|
|
|
<Link to="/" class="button plain">
|
2024-08-13 15:26:23 +08:00
|
|
|
<Icon icon="home" size="l" alt={t`Home`} />
|
2024-03-23 01:07:24 +08:00
|
|
|
</Link>
|
|
|
|
</div>
|
2024-08-13 15:26:23 +08:00
|
|
|
<h1>
|
|
|
|
<Trans>Filters</Trans>
|
|
|
|
</h1>
|
2024-03-23 01:07:24 +08:00
|
|
|
<div class="header-side">
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="plain"
|
|
|
|
onClick={() => {
|
|
|
|
setShowFiltersAddEditModal(true);
|
|
|
|
}}
|
|
|
|
>
|
2024-08-13 15:26:23 +08:00
|
|
|
<Icon icon="plus" size="l" alt={t`New filter`} />
|
2024-03-23 01:07:24 +08:00
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</header>
|
|
|
|
<main>
|
|
|
|
{filters.length > 0 ? (
|
|
|
|
<>
|
|
|
|
<ul class="filters-list">
|
|
|
|
{filters.map((filter) => {
|
|
|
|
const { id, title, expiresAt, keywords } = filter;
|
|
|
|
return (
|
|
|
|
<li key={id}>
|
|
|
|
<div>
|
|
|
|
<h2>{title}</h2>
|
|
|
|
{keywords?.length > 0 && (
|
|
|
|
<div>
|
|
|
|
{keywords.map((k) => (
|
|
|
|
<>
|
|
|
|
<span class="tag collapsed insignificant">
|
|
|
|
{k.wholeWord ? `“${k.keyword}”` : k.keyword}
|
|
|
|
</span>{' '}
|
|
|
|
</>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<small class="insignificant">
|
|
|
|
<ExpiryStatus expiresAt={expiresAt} />
|
|
|
|
</small>
|
|
|
|
</div>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="plain"
|
|
|
|
onClick={() => {
|
|
|
|
setShowFiltersAddEditModal({
|
|
|
|
filter,
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Icon icon="pencil" size="l" alt="Edit filter" />
|
|
|
|
</button>
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</ul>
|
|
|
|
{filters.length > 1 && (
|
|
|
|
<footer class="ui-state">
|
|
|
|
<small class="insignificant">
|
2024-08-13 15:26:23 +08:00
|
|
|
<Plural
|
|
|
|
value={filters.length}
|
|
|
|
one="# filter"
|
|
|
|
other="# filters"
|
|
|
|
/>
|
2024-03-23 01:07:24 +08:00
|
|
|
</small>
|
|
|
|
</footer>
|
|
|
|
)}
|
|
|
|
</>
|
|
|
|
) : uiState === 'loading' ? (
|
|
|
|
<p class="ui-state">
|
|
|
|
<Loader />
|
|
|
|
</p>
|
|
|
|
) : uiState === 'error' ? (
|
2024-08-13 15:26:23 +08:00
|
|
|
<p class="ui-state">
|
|
|
|
<Trans>Unable to load filters.</Trans>
|
|
|
|
</p>
|
2024-03-23 01:07:24 +08:00
|
|
|
) : (
|
2024-08-13 15:26:23 +08:00
|
|
|
<p class="ui-state">
|
|
|
|
<Trans>No filters yet.</Trans>
|
|
|
|
</p>
|
2024-03-23 01:07:24 +08:00
|
|
|
)}
|
|
|
|
</main>
|
|
|
|
</div>
|
|
|
|
{!!showFiltersAddEditModal && (
|
|
|
|
<Modal
|
2024-08-13 15:26:23 +08:00
|
|
|
title={t`Add filter`}
|
2024-03-23 01:07:24 +08:00
|
|
|
onClose={() => {
|
|
|
|
setShowFiltersAddEditModal(false);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<FiltersAddEdit
|
|
|
|
filter={showFiltersAddEditModal?.filter}
|
|
|
|
onClose={(result) => {
|
|
|
|
if (result.state === 'success') {
|
|
|
|
reload();
|
|
|
|
}
|
|
|
|
setShowFiltersAddEditModal(false);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</Modal>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-03-26 23:45:22 +08:00
|
|
|
let _id = 1;
|
|
|
|
const incID = () => _id++;
|
2024-03-23 01:07:24 +08:00
|
|
|
function FiltersAddEdit({ filter, onClose }) {
|
2024-08-13 15:26:23 +08:00
|
|
|
const { _ } = useLingui();
|
2024-03-23 01:07:24 +08:00
|
|
|
const { masto } = api();
|
|
|
|
const [uiState, setUIState] = useState('default');
|
|
|
|
const editMode = !!filter;
|
|
|
|
const { context, expiresAt, id, keywords, title, filterAction } =
|
|
|
|
filter || {};
|
|
|
|
const hasExpiry = !!expiresAt;
|
|
|
|
const expiresAtDate = hasExpiry && new Date(expiresAt);
|
|
|
|
const [editKeywords, setEditKeywords] = useState(keywords || []);
|
|
|
|
const keywordsRef = useRef();
|
|
|
|
|
|
|
|
// Hacky way of handling removed keywords for both existing and new ones
|
|
|
|
const [removedKeywordIDs, setRemovedKeywordIDs] = useState([]);
|
2024-03-26 23:45:22 +08:00
|
|
|
const [removedKeyword_IDs, setRemovedKeyword_IDs] = useState([]);
|
|
|
|
|
|
|
|
const filteredEditKeywords = editKeywords.filter(
|
|
|
|
(k) =>
|
|
|
|
!removedKeywordIDs.includes(k.id) && !removedKeyword_IDs.includes(k._id),
|
|
|
|
);
|
2024-03-23 01:07:24 +08:00
|
|
|
|
|
|
|
return (
|
|
|
|
<div class="sheet" id="filters-add-edit-modal">
|
|
|
|
{!!onClose && (
|
|
|
|
<button type="button" class="sheet-close" onClick={onClose}>
|
2024-08-13 15:26:23 +08:00
|
|
|
<Icon icon="x" alt={t`Close`} />
|
2024-03-23 01:07:24 +08:00
|
|
|
</button>
|
|
|
|
)}
|
|
|
|
<header>
|
2024-08-13 15:26:23 +08:00
|
|
|
<h2>{editMode ? t`Edit filter` : t`New filter`}</h2>
|
2024-03-23 01:07:24 +08:00
|
|
|
</header>
|
|
|
|
<main>
|
|
|
|
<form
|
|
|
|
onSubmit={(e) => {
|
|
|
|
e.preventDefault();
|
|
|
|
const formData = new FormData(e.target);
|
|
|
|
const title = formData.get('title');
|
|
|
|
const keywordIDs = formData.getAll('keyword_attributes[][id]');
|
|
|
|
const keywordKeywords = formData.getAll(
|
|
|
|
'keyword_attributes[][keyword]',
|
|
|
|
);
|
|
|
|
// const keywordWholeWords = formData.getAll(
|
|
|
|
// 'keyword_attributes[][whole_word]',
|
|
|
|
// );
|
|
|
|
// Not using getAll because it skips the empty checkboxes
|
|
|
|
const keywordWholeWords = [
|
|
|
|
...keywordsRef.current.querySelectorAll(
|
|
|
|
'input[name="keyword_attributes[][whole_word]"]',
|
|
|
|
),
|
|
|
|
].map((i) => i.checked);
|
|
|
|
const keywordsAttributes = keywordKeywords.map((k, i) => ({
|
|
|
|
id: keywordIDs[i] || undefined,
|
|
|
|
keyword: k,
|
|
|
|
wholeWord: keywordWholeWords[i],
|
|
|
|
}));
|
|
|
|
// if (editMode && keywords?.length) {
|
|
|
|
// // Find which one got deleted and add to keywordsAttributes
|
|
|
|
// keywords.forEach((k) => {
|
|
|
|
// if (!keywordsAttributes.find((ka) => ka.id === k.id)) {
|
|
|
|
// keywordsAttributes.push({
|
|
|
|
// ...k,
|
|
|
|
// _destroy: true,
|
|
|
|
// });
|
|
|
|
// }
|
|
|
|
// });
|
|
|
|
// }
|
|
|
|
if (editMode && removedKeywordIDs?.length) {
|
|
|
|
removedKeywordIDs.forEach((id) => {
|
|
|
|
keywordsAttributes.push({
|
|
|
|
id,
|
|
|
|
_destroy: true,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
const context = formData.getAll('context');
|
|
|
|
let expiresIn = formData.get('expires_in');
|
|
|
|
const filterAction = formData.get('filter_action');
|
|
|
|
console.log({
|
|
|
|
title,
|
|
|
|
keywordIDs,
|
|
|
|
keywords: keywordKeywords,
|
|
|
|
wholeWords: keywordWholeWords,
|
|
|
|
keywordsAttributes,
|
|
|
|
context,
|
|
|
|
expiresIn,
|
|
|
|
filterAction,
|
|
|
|
});
|
|
|
|
|
|
|
|
// Required fields
|
|
|
|
if (!title || !context?.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
setUIState('loading');
|
|
|
|
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
let filterResult;
|
|
|
|
|
|
|
|
if (editMode) {
|
|
|
|
if (expiresIn === '' || expiresIn === null) {
|
|
|
|
// No value
|
|
|
|
// Preserve existing expiry if not specified
|
|
|
|
// Seconds from now to expiresAtDate
|
|
|
|
// Other clients don't do this
|
2024-04-26 09:58:07 +08:00
|
|
|
if (hasExpiry) {
|
|
|
|
expiresIn = Math.floor(
|
|
|
|
(expiresAtDate - new Date()) / 1000,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
expiresIn = null;
|
|
|
|
}
|
2024-03-23 01:07:24 +08:00
|
|
|
} else if (expiresIn === '0' || expiresIn === 0) {
|
|
|
|
// 0 = Never
|
|
|
|
expiresIn = null;
|
|
|
|
} else {
|
|
|
|
expiresIn = +expiresIn;
|
|
|
|
}
|
|
|
|
filterResult = await masto.v2.filters.$select(id).update({
|
|
|
|
title,
|
|
|
|
context,
|
|
|
|
expiresIn,
|
|
|
|
keywordsAttributes,
|
|
|
|
filterAction,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
expiresIn = +expiresIn || null;
|
|
|
|
filterResult = await masto.v2.filters.create({
|
|
|
|
title,
|
|
|
|
context,
|
|
|
|
expiresIn,
|
|
|
|
keywordsAttributes,
|
|
|
|
filterAction,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
console.log({ filterResult });
|
|
|
|
setUIState('default');
|
|
|
|
onClose?.({
|
|
|
|
state: 'success',
|
|
|
|
filter: filterResult,
|
|
|
|
});
|
|
|
|
} catch (error) {
|
|
|
|
console.error(error);
|
|
|
|
setUIState('error');
|
|
|
|
alert(
|
|
|
|
editMode
|
2024-08-13 15:26:23 +08:00
|
|
|
? t`Unable to edit filter`
|
|
|
|
: t`Unable to create filter`,
|
2024-03-23 01:07:24 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<div class="filter-form-row">
|
|
|
|
<label>
|
2024-08-13 15:26:23 +08:00
|
|
|
<b>
|
|
|
|
<Trans>Title</Trans>
|
|
|
|
</b>
|
2024-03-23 01:07:24 +08:00
|
|
|
<input
|
|
|
|
type="text"
|
|
|
|
name="title"
|
|
|
|
defaultValue={title}
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
dir="auto"
|
|
|
|
required
|
|
|
|
/>
|
|
|
|
</label>
|
|
|
|
</div>
|
|
|
|
<div class="filter-form-keywords" ref={keywordsRef}>
|
2024-03-26 23:45:22 +08:00
|
|
|
{filteredEditKeywords.length ? (
|
2024-03-23 01:07:24 +08:00
|
|
|
<ul class="filter-keywords">
|
2024-03-26 23:45:22 +08:00
|
|
|
{filteredEditKeywords.map((k) => {
|
|
|
|
const { id, keyword, wholeWord, _id } = k;
|
2024-03-23 01:07:24 +08:00
|
|
|
return (
|
2024-03-26 23:45:22 +08:00
|
|
|
<li key={`${id}-${_id}`}>
|
2024-03-23 01:07:24 +08:00
|
|
|
<input
|
|
|
|
type="hidden"
|
|
|
|
name="keyword_attributes[][id]"
|
|
|
|
value={id}
|
|
|
|
/>
|
|
|
|
<input
|
|
|
|
name="keyword_attributes[][keyword]"
|
|
|
|
type="text"
|
|
|
|
defaultValue={keyword}
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
required
|
2024-08-04 13:32:30 +08:00
|
|
|
dir="auto"
|
2024-03-23 01:07:24 +08:00
|
|
|
/>
|
|
|
|
<div class="filter-keyword-actions">
|
|
|
|
<label>
|
|
|
|
<input
|
|
|
|
name="keyword_attributes[][whole_word]"
|
|
|
|
type="checkbox"
|
|
|
|
value={id} // Hacky way to map checkbox boolean to the keyword id
|
|
|
|
defaultChecked={wholeWord}
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
/>{' '}
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>Whole word</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
</label>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="light danger small"
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
onClick={() => {
|
|
|
|
if (id) {
|
|
|
|
removedKeywordIDs.push(id);
|
|
|
|
setRemovedKeywordIDs([...removedKeywordIDs]);
|
2024-03-26 23:45:22 +08:00
|
|
|
} else if (_id) {
|
|
|
|
removedKeyword_IDs.push(_id);
|
|
|
|
setRemovedKeyword_IDs([...removedKeyword_IDs]);
|
2024-03-23 01:07:24 +08:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
2024-08-13 15:26:23 +08:00
|
|
|
<Icon icon="x" alt={t`Remove`} />
|
2024-03-23 01:07:24 +08:00
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</ul>
|
|
|
|
) : (
|
|
|
|
<div class="filter-keywords">
|
2024-08-13 15:26:23 +08:00
|
|
|
<div class="insignificant">
|
|
|
|
<Trans>No keywords. Add one.</Trans>
|
|
|
|
</div>
|
2024-03-23 01:07:24 +08:00
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<footer class="filter-keywords-footer">
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="light"
|
|
|
|
onClick={() => {
|
|
|
|
setEditKeywords([
|
|
|
|
...editKeywords,
|
|
|
|
{
|
2024-03-26 23:45:22 +08:00
|
|
|
_id: incID(),
|
2024-03-23 01:07:24 +08:00
|
|
|
keyword: '',
|
|
|
|
wholeWord: true,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
setTimeout(() => {
|
|
|
|
// Focus last input
|
|
|
|
const fields =
|
|
|
|
keywordsRef.current.querySelectorAll(
|
|
|
|
'input[type="text"]',
|
|
|
|
);
|
|
|
|
fields[fields.length - 1]?.focus?.();
|
|
|
|
}, 10);
|
|
|
|
}}
|
|
|
|
>
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>Add keyword</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
</button>{' '}
|
2024-03-26 23:45:22 +08:00
|
|
|
{filteredEditKeywords?.length > 1 && (
|
2024-03-23 01:07:24 +08:00
|
|
|
<small class="insignificant">
|
2024-08-13 15:26:23 +08:00
|
|
|
<Plural
|
|
|
|
value={filteredEditKeywords.length}
|
|
|
|
one="# keyword"
|
|
|
|
other="# keywords"
|
|
|
|
/>
|
2024-03-23 01:07:24 +08:00
|
|
|
</small>
|
|
|
|
)}
|
|
|
|
</footer>
|
|
|
|
</div>
|
|
|
|
<div class="filter-form-cols">
|
|
|
|
<div class="filter-form-col">
|
|
|
|
<div>
|
2024-08-13 15:26:23 +08:00
|
|
|
<b>
|
|
|
|
<Trans>Filter from…</Trans>
|
|
|
|
</b>
|
2024-03-23 01:07:24 +08:00
|
|
|
</div>
|
|
|
|
{FILTER_CONTEXT.map((ctx) => (
|
|
|
|
<div>
|
|
|
|
<label
|
|
|
|
class={
|
|
|
|
FILTER_CONTEXT_UNIMPLEMENTED.includes(ctx)
|
|
|
|
? 'insignificant'
|
|
|
|
: ''
|
|
|
|
}
|
|
|
|
>
|
|
|
|
<input
|
|
|
|
type="checkbox"
|
|
|
|
name="context"
|
|
|
|
value={ctx}
|
|
|
|
defaultChecked={!!context ? context.includes(ctx) : true}
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
/>{' '}
|
2024-08-13 15:26:23 +08:00
|
|
|
{_(FILTER_CONTEXT_LABELS[ctx])}
|
2024-03-23 01:07:24 +08:00
|
|
|
{FILTER_CONTEXT_UNIMPLEMENTED.includes(ctx) ? '*' : ''}
|
|
|
|
</label>{' '}
|
|
|
|
</div>
|
|
|
|
))}
|
|
|
|
<p>
|
2024-08-13 15:26:23 +08:00
|
|
|
<small class="insignificant">
|
|
|
|
<Trans>* Not implemented yet</Trans>
|
|
|
|
</small>
|
2024-03-23 01:07:24 +08:00
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
<div class="filter-form-col">
|
|
|
|
{editMode && (
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
Status:{' '}
|
|
|
|
<b>
|
|
|
|
<ExpiryStatus expiresAt={expiresAt} showNeverExpires />
|
|
|
|
</b>
|
2024-08-13 15:26:23 +08:00
|
|
|
</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
)}
|
|
|
|
<div>
|
|
|
|
<label for="filters-expires_in">
|
2024-08-13 15:26:23 +08:00
|
|
|
{editMode ? t`Change expiry` : t`Expiry`}
|
2024-03-23 01:07:24 +08:00
|
|
|
</label>
|
|
|
|
<select
|
|
|
|
id="filters-expires_in"
|
|
|
|
name="expires_in"
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
defaultValue={editMode ? undefined : 0}
|
|
|
|
>
|
|
|
|
{editMode && <option></option>}
|
|
|
|
{EXPIRY_DURATIONS.map((v) => (
|
2024-08-13 15:26:23 +08:00
|
|
|
<option value={v}>
|
|
|
|
{typeof EXPIRY_DURATIONS_LABELS[v] === 'function'
|
|
|
|
? EXPIRY_DURATIONS_LABELS[v]()
|
|
|
|
: _(EXPIRY_DURATIONS_LABELS[v])}
|
|
|
|
</option>
|
2024-03-23 01:07:24 +08:00
|
|
|
))}
|
|
|
|
</select>
|
|
|
|
</div>
|
|
|
|
<p>
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>Filtered post will be…</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
<br />
|
|
|
|
<label class="ib">
|
|
|
|
<input
|
|
|
|
type="radio"
|
|
|
|
name="filter_action"
|
|
|
|
value="warn"
|
|
|
|
defaultChecked={filterAction === 'warn' || !editMode}
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
/>{' '}
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>minimized</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
</label>{' '}
|
|
|
|
<label class="ib">
|
|
|
|
<input
|
|
|
|
type="radio"
|
|
|
|
name="filter_action"
|
|
|
|
value="hide"
|
|
|
|
defaultChecked={filterAction === 'hide'}
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
/>{' '}
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>hidden</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
</label>
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<footer class="filter-form-footer">
|
|
|
|
<span>
|
|
|
|
<button type="submit" disabled={uiState === 'loading'}>
|
2024-08-13 15:26:23 +08:00
|
|
|
{editMode ? t`Save` : t`Create`}
|
2024-03-23 01:07:24 +08:00
|
|
|
</button>{' '}
|
|
|
|
<Loader abrupt hidden={uiState !== 'loading'} />
|
|
|
|
</span>
|
|
|
|
{editMode && (
|
|
|
|
<MenuConfirm
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
align="end"
|
|
|
|
menuItemClassName="danger"
|
2024-08-13 15:26:23 +08:00
|
|
|
confirmLabel={t`Delete this filter?`}
|
2024-03-23 01:07:24 +08:00
|
|
|
onClick={() => {
|
|
|
|
setUIState('loading');
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
await masto.v2.filters.$select(id).remove();
|
|
|
|
setUIState('default');
|
|
|
|
onClose?.({
|
|
|
|
state: 'success',
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
setUIState('error');
|
2024-08-13 15:26:23 +08:00
|
|
|
alert(t`Unable to delete filter.`);
|
2024-03-23 01:07:24 +08:00
|
|
|
}
|
|
|
|
})();
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="light danger"
|
|
|
|
onClick={() => {}}
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
>
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>Delete…</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
</button>
|
|
|
|
</MenuConfirm>
|
|
|
|
)}
|
|
|
|
</footer>
|
|
|
|
</form>
|
|
|
|
</main>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function ExpiryStatus({ expiresAt, showNeverExpires }) {
|
|
|
|
const hasExpiry = !!expiresAt;
|
|
|
|
const expiresAtDate = hasExpiry && new Date(expiresAt);
|
|
|
|
const expired = hasExpiry && expiresAtDate <= new Date();
|
|
|
|
|
|
|
|
// If less than a minute left, re-render interval every second, else every minute
|
|
|
|
const [_, rerender] = useReducer((c) => c + 1, 0);
|
|
|
|
useInterval(rerender, expired || 30_000);
|
|
|
|
|
|
|
|
return expired ? (
|
2024-08-13 15:26:23 +08:00
|
|
|
t`Expired`
|
2024-03-23 01:07:24 +08:00
|
|
|
) : hasExpiry ? (
|
2024-08-13 15:26:23 +08:00
|
|
|
<Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
Expiring <RelativeTime datetime={expiresAtDate} />
|
2024-08-13 15:26:23 +08:00
|
|
|
</Trans>
|
2024-03-23 01:07:24 +08:00
|
|
|
) : (
|
2024-08-13 15:26:23 +08:00
|
|
|
showNeverExpires && t`Never expires`
|
2024-03-23 01:07:24 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default Filters;
|